一文详解Java如何自动生成简单的Mermaid类图

 更新时间:2025年10月10日 08:32:12   作者:金銀銅鐵  
这篇文章主要为大家详细介绍了Java如何自动生成简单的Mermaid类图,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下

背景

虽说手动生成类图的过程有利于加深自己的理解,但是查看各个类/接口的信息毕竟比较麻烦,如果可以把生成类图的过程自动化,就可以大大提升画类图的效率了。

本文展示了我自己写的可以生成简单类图的 java 代码。文中展示了用它生成的以下类的类图

  • ArrayList
  • LinkedList
  • java.util.Arrays$ArrayListArrays.asList(...) 方法返回的是它的实例)
  • List/Set/Deque
  • HashMap/LinkedHashMap/ConcurrentHashMap
  • Set12/SetNSet.of(...) 方法返回的是它们的实例)
  • java.util.JumboEnumSet 和 java.util.RegularEnumSet (EnumSet.of(...) 方法返回的是它们的实例)
  • java.util.concurrent.ThreadPoolExecutor

代码

我写了如下的 java 代码,它可以自动生成简单的 Mermaid 类图。 请将以下代码保存为 ClassDiagramGenerator.java 

import java.lang.reflect.AccessFlag;
import java.util.*;

public class ClassDiagramGenerator {
    private final Map<Class<?>, Class<?>[]> realizationRelations = new HashMap<>();
    private final Map<Class<?>, Class<?>> inheritanceRelations = new HashMap<>();

    private final Set<Class<?>> analyzedClasses = new HashSet<>();

    public static void main(String[] args) {
        ClassDiagramGenerator generator = new ClassDiagramGenerator();
        generator.convert(args).forEach(generator::analyzeHierarchy);

        generator.generateClassDiagram();
        generator.generateNameMappingTable();
    }

    /**
     * Convert class name to the corresponding class object
     *
     * @param classNames give class names
     * @return a list that contains corresponding class objects
     */
    private List<Class<?>> convert(String[] classNames) {
        if (classNames.length == 0) {
            String hint = "Please refer to below usage and specify at least ONE class name!";
            String usage = "Usage: java ClassDiagramGenerator 'java.util.ArrayList' 'java.util.LinkedList'";
            throw new IllegalArgumentException(String.join(System.lineSeparator(), hint, usage));
        }

        List<Class<?>> classList = new ArrayList<>(classNames.length);
        for (String className : classNames) {
            try {
                Class<?> clazz = Class.forName(className);
                classList.add(clazz);
            } catch (ClassNotFoundException e) {
                String message = String.format("Class with name=%s can't be found, please check!", className);
                throw new RuntimeException(message);
            }
        }
        return classList;
    }

    /**
     * Generate header for Mermaid class diagram
     */
    private void generateHeader() {
        System.out.println("```mermaid");
        System.out.println("classDiagram");
        System.out.println();
    }

    /**
     * Generate footer for Mermaid class diagram
     */
    private void generateFooter() {
        System.out.println("```");
    }

    /**
     * Generate main content in Mermaid class diagram
     */
    private void generateClassDiagram() {
        generateHeader();
        doGenerateClassDiagram();
        generateFooter();
    }

    /**
     * Generate a Markdown table that contains name mapping
     */
    private void generateNameMappingTable() {
        System.out.println();
        System.out.println("| 在上图中的类名/接口名 | `Fully Qualified Name` |");
        System.out.println("| --- | --- |");
        Map<String, String> classNames = new TreeMap<>();
        analyzedClasses.forEach(c -> {
            String simpleName = c.getSimpleName();
            if (classNames.containsKey(simpleName)) {
                String prevName = classNames.get(simpleName);
                String currName = c.getName();
                String message = String.format("Duplicated simple class name detected! (%s and %s have the same simple name)", prevName, currName);
                throw new IllegalArgumentException(message);
            }
            classNames.put(simpleName, c.getName());
        });

        classNames.forEach((simpleName, name) -> {
            String row = String.format("| `%s` | `%s` |", simpleName, name);
            System.out.println(row);
        });
    }

    private void doGenerateClassDiagram() {
        analyzedClasses.forEach(c -> {
            if (inheritanceRelations.containsKey(c)) {
                System.out.printf("%s <|-- %s%n", inheritanceRelations.get(c).getSimpleName(), c.getSimpleName());
            }
            if (realizationRelations.containsKey(c)) {
                String type = c.isInterface() ? "<|--" : "<|..";
                Arrays.stream(realizationRelations.get(c)).forEach(item -> {
                    System.out.printf("%s %s %s%n", item.getSimpleName(), type, c.getSimpleName());
                });
            }
        });

        generateSpecialClassAnnotation();
    }

    /**
     * This method generates annotation for
     * 1. Abstract classes
     * 2. Interfaces
     */
    private void generateSpecialClassAnnotation() {
        Set<Class<?>> abstractClasses = new LinkedHashSet<>();
        Set<Class<?>> interfaces = new LinkedHashSet<>();

        analyzedClasses.forEach(c -> {
            if (c.isInterface()) {
                interfaces.add(c);
            } else if (c.accessFlags().contains(AccessFlag.ABSTRACT)) {
                abstractClasses.add(c);
            }
        });

        if (!abstractClasses.isEmpty() || !interfaces.isEmpty()) {
            System.out.println();
            abstractClasses.forEach(c -> System.out.println("<<Abstract>> " + c.getSimpleName()));
            interfaces.forEach(c -> System.out.println("<<interface>> " + c.getSimpleName()));
        }
    }

    private void analyzeHierarchy(Class<?> currClass) {
        if (!analyzedClasses.contains(currClass)) {
            analyzeSuperClass(currClass);
            analyzeInterfaces(currClass);

            analyzedClasses.add(currClass);
        }
    }

    private void analyzeSuperClass(Class<?> currClass) {
        Class<?> superclass = currClass.getSuperclass();
        if (superclass == null || superclass == Object.class) {
            return;
        }
        analyzeHierarchy(superclass);
        inheritanceRelations.put(currClass, superclass);
    }

    private void analyzeInterfaces(Class<?> currClass) {
        Class<?>[] interfaces = currClass.getInterfaces();
        for (Class<?> item : interfaces) {
            analyzeHierarchy(item);
        }

        realizationRelations.put(currClass, interfaces);
    }
}

用以下命令可以编译 ClassDiagramGenerator.java

javac ClassDiagramGenerator.java

注意事项

请注意,ClassDiagramGenerator 生成的类图中,不包含任何泛型信息,而且也不展示任何字段/方法。

例子

下面举一些例子,说明 ClassDiagramGenerator.java 的用法。

例1: 生成ArrayList的类图

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.ArrayList'

运行结果是 Markdown 格式的

运行结果

在上图中的类名/接口名Fully Qualified Name
AbstractCollectionjava.util.AbstractCollection
AbstractListjava.util.AbstractList
ArrayListjava.util.ArrayList
Cloneablejava.lang.Cloneable
Collectionjava.util.Collection
Iterablejava.lang.Iterable
Listjava.util.List
RandomAccessjava.util.RandomAccess
SequencedCollectionjava.util.SequencedCollection
Serializablejava.io.Serializable

变通的方式

如果您无法在掘金的文档中使用 Mermaid,那么也可以前往 mermaid.live/ 来查看对应的类图,我在 mermaid.live/ 看到的效果如下 (需要自行将结果复制到那个网页去)

例2: 生成LinkedList的类图

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.LinkedList'

运行结果是 Markdown 格式的,展示如下

运行结果

在上图中的类名/接口名Fully Qualified Name
AbstractCollectionjava.util.AbstractCollection
AbstractListjava.util.AbstractList
AbstractSequentialListjava.util.AbstractSequentialList
Cloneablejava.lang.Cloneable
Collectionjava.util.Collection
Dequejava.util.Deque
Iterablejava.lang.Iterable
LinkedListjava.util.LinkedList
Listjava.util.List
Queuejava.util.Queue
SequencedCollectionjava.util.SequencedCollection
Serializablejava.io.Serializable

例3: 生成java.util.Arrays$ArrayList的类图

调用 Arrays.asList(...) 方法,得到的是 java.util.Arrays$ArrayList 的实例。 请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.Arrays$ArrayList'

运行结果是 Markdown 格式的,展示如下

运行结果

请注意下图中的 ArrayListjava.util.Arrays 中的一个嵌套类,而不是我们平时常用的 java.util.ArrayList

在上图中的类名/接口名Fully Qualified Name
AbstractCollectionjava.util.AbstractCollection
AbstractListjava.util.AbstractList
ArrayListjava.util.Arrays$ArrayList
Collectionjava.util.Collection
Iterablejava.lang.Iterable
Listjava.util.List
RandomAccessjava.util.RandomAccess
SequencedCollectionjava.util.SequencedCollection
Serializablejava.io.Serializable

例4: 生成List/Set/Deque的类图

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.List' 'java.util.Set' 'java.util.Deque'

运行结果是 Markdown 格式的,展示如下

运行结果

在上图中的类名/接口名Fully Qualified Name
Collectionjava.util.Collection
Dequejava.util.Deque
Iterablejava.lang.Iterable
Listjava.util.List
Queuejava.util.Queue
SequencedCollectionjava.util.SequencedCollection
Setjava.util.Set

例5: 生成HashMap/LinkedHashMap/ConcurrentHashMap的类图

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.HashMap' 'java.util.LinkedHashMap' 'java.util.concurrent.ConcurrentHashMap'

运行结果是 Markdown 格式的,展示如下

运行结果

在上图中的类名/接口名Fully Qualified Name
AbstractMapjava.util.AbstractMap
Cloneablejava.lang.Cloneable
ConcurrentHashMapjava.util.concurrent.ConcurrentHashMap
ConcurrentMapjava.util.concurrent.ConcurrentMap
HashMapjava.util.HashMap
LinkedHashMapjava.util.LinkedHashMap
Mapjava.util.Map
SequencedMapjava.util.SequencedMap
Serializablejava.io.Serializable

例6: 生成java.util.ImmutableCollections$Set12和java.util.ImmutableCollections$SetN的类图

当我们调用 Set.of(...) 方法时,会得到以下两个类的实例 

  • java.util.ImmutableCollections$Set12
  • java.util.ImmutableCollections$SetN

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.ImmutableCollections$Set12' 'java.util.ImmutableCollections$SetN'

运行结果是 Markdown 格式的,展示如下

运行结果

在上图中的类名/接口名Fully Qualified Name
AbstractCollectionjava.util.AbstractCollection
AbstractImmutableCollectionjava.util.ImmutableCollections$AbstractImmutableCollection
AbstractImmutableSetjava.util.ImmutableCollections$AbstractImmutableSet
Collectionjava.util.Collection
Iterablejava.lang.Iterable
Serializablejava.io.Serializable
Setjava.util.Set
Set12java.util.ImmutableCollections$Set12
SetNjava.util.ImmutableCollections$SetN

例7: 生成java.util.JumboEnumSet和java.util.RegularEnumSet的类图

当我们调用 EnumSet.of(...) 方法时,会得到以下两个类的实例 

  • java.util.JumboEnumSet
  • java.util.RegularEnumSet

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.JumboEnumSet' 'java.util.RegularEnumSet'

运行结果是 Markdown 格式的,展示如下

运行结果

在上图中的类名/接口名Fully Qualified Name
AbstractCollectionjava.util.AbstractCollection
AbstractSetjava.util.AbstractSet
Cloneablejava.lang.Cloneable
Collectionjava.util.Collection
EnumSetjava.util.EnumSet
Iterablejava.lang.Iterable
JumboEnumSetjava.util.JumboEnumSet
RegularEnumSetjava.util.RegularEnumSet
Serializablejava.io.Serializable
Setjava.util.Set

例8: 生成ThreadPoolExecutor的类图

请运行以下命令以生成对应的内容 

java ClassDiagramGenerator 'java.util.concurrent.ThreadPoolExecutor'

运行结果是 Markdown 格式的,展示如下

运行结果

在上图中的类名/接口名Fully Qualified Name
AbstractExecutorServicejava.util.concurrent.AbstractExecutorService
AutoCloseablejava.lang.AutoCloseable
Executorjava.util.concurrent.Executor
ExecutorServicejava.util.concurrent.ExecutorService
ThreadPoolExecutorjava.util.concurrent.ThreadPoolExecutor

到此这篇关于一文详解Java如何自动生成简单的Mermaid类图的文章就介绍到这了,更多相关Java生成Mermaid类图内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java利用for循环输出空心菱形的实例代码

    Java利用for循环输出空心菱形的实例代码

    这篇文章主要介绍了Java利用for循环输出空心菱形的实例代码,需要的朋友可以参考下
    2014-02-02
  • SpringBoot项目部署到阿里云服务器的实现步骤

    SpringBoot项目部署到阿里云服务器的实现步骤

    本文主要介绍了SpringBoot项目部署到阿里云服务器的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-06-06
  • 详解Java的回调机制

    详解Java的回调机制

    最近学习java,接触到了回调机制(CallBack)。初识时感觉比较混乱,而且在网上搜索到的相关的讲解,本文介绍了Java的回调机制,有兴趣的同学可以了解一下
    2016-10-10
  • 利用Spring Data MongoDB持久化文档数据的方法教程

    利用Spring Data MongoDB持久化文档数据的方法教程

    这篇文章主要给大家介绍了关于利用Spring Data MongoDB持久化文档数据的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-08-08
  • 基于spring boot 2和shiro实现身份验证案例

    基于spring boot 2和shiro实现身份验证案例

    这篇文章主要介绍了基于spring boot 2和shiro实现身份验证案例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • LeetCode 动态规划之矩阵区域和详情

    LeetCode 动态规划之矩阵区域和详情

    这篇文章主要介绍了LeetCode 动态规划之矩阵区域和详情,文章基于Java的相关资料展开对LeetCode 动态规划的详细介绍,需要的小伙伴可以参考一下
    2022-04-04
  • Java springboot探究配置文件优先级

    Java springboot探究配置文件优先级

    在springboot项目中,我们可以通过在yml文件中设置变量,再通过@Value注解来获得这个变量并使用,但如果这个项目已经部署到服务器上,我们想更改这个数据了需要怎么做呢,其实在springboot项目中,配置文件是有优先级的
    2023-04-04
  • Java实战个人博客系统的实现流程

    Java实战个人博客系统的实现流程

    读万卷书不如行万里路,只学书上的理论是远远不够的,只有在实战中才能获得能力的提升,本篇文章手把手带你用java+springboot+mybatis+redis+vue+elementui+Mysql实现一个个人博客系统,大家可以在过程中查缺补漏,提升水平
    2022-01-01
  • LeetCode程序员面试题之无重复字符的最长子串

    LeetCode程序员面试题之无重复字符的最长子串

    Java计算无重复字符的最长子串是一种常见的字符串处理算法,它的目的是找出一个字符串中无重复字符的最长子串。该算法可以很好地解决一些字符串处理问题,比如寻找字符串中重复字符的位置,以及计算字符串中无重复字符的最长子串的长度。
    2023-02-02
  • SpringBoot + proguard+maven多模块实现代码混淆的方法

    SpringBoot + proguard+maven多模块实现代码混淆的方法

    这篇文章主要介绍了SpringBoot + proguard+maven多模块实现代码混淆的方法,多模块跟单模块一样,在需要混淆模块的pom文件中加入proguard依赖及配置,本文给大家讲解的非常详细,感兴趣的朋友一起看看吧
    2024-02-02

最新评论