为指定 java 类生成 PlantUML puml文件工具( v2 )

AttributePumlVO.java:
import lombok.Getter;
import lombok.Setter;import java.io.Serializable;@Getter
@Setter
public class AttributePumlVO implements Serializable {/*** 属性名称*/private String name;/*** 属性类型*/private Class type;@Overridepublic String toString() {return "\ticon_hammer " + this.name + ": " + this.type.getSimpleName() + "\n";}
}

ClassPumlGenerate.java:

import lombok.Getter;
import lombok.Setter;
import org.reflections.Reflections;import java.io.*;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLDecoder;
import java.util.*;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;@Getter
@Setter
public class ClassPumlGenerate {private Set<String> classIdentifiers = new HashSet<>();private List<ClassPumlVO> classPumlList = new ArrayList<>();private static final Set<String> JDK_METHOD_NAMES = new HashSet<>();private static final Set<String> JDK_CLASS_NAMES = new HashSet<>();private static final Set<String> JDK_ATTRIBUTE_NAMES = new HashSet<>();static {JDK_METHOD_NAMES.add( "wait" );JDK_METHOD_NAMES.add( "equals" );JDK_METHOD_NAMES.add( "toString" );JDK_METHOD_NAMES.add( "hashCode" );JDK_METHOD_NAMES.add( "notify" );JDK_METHOD_NAMES.add( "notifyAll" );JDK_METHOD_NAMES.add( "finalize" );JDK_CLASS_NAMES.add( "boolean" );JDK_CLASS_NAMES.add( "void" );JDK_CLASS_NAMES.add( "int" );JDK_CLASS_NAMES.add( "long" );JDK_CLASS_NAMES.add( "float" );JDK_CLASS_NAMES.add( "byte" );JDK_CLASS_NAMES.add( "double" );JDK_CLASS_NAMES.add( "short" );JDK_CLASS_NAMES.add( "[Ljava.lang.Object;" );JDK_CLASS_NAMES.add( "[B" );JDK_CLASS_NAMES.add( "[Ljava.lang.String;" );JDK_ATTRIBUTE_NAMES.add( "serialVersionUID" );}public void generatePumlForPackage( String packagePath,String outputPath,boolean ignoreInterface,boolean ignoreProperties ){BufferedWriter writer = null;try {writer = new BufferedWriter(new FileWriter( outputPath ));this.classPumlList = new ArrayList<>();List<Class<?>> clazzList = this.getClasses(packagePath);for( Class clazz:clazzList ){this.generate( clazz,ignoreInterface,ignoreProperties);}writer.write( "@startuml\r\n" );writer.write( "!define icon_hammer <img:C:\\E\\sucai\\banshou3.png>\r\n" );writer.write( "!define icon_cube <img:C:\\E\\sucai\\cube_3.png>\r\n" );writer.write( "skinparam Class {\r\n" );writer.write( "\tBackgroundColor  #d3dcef/white\r\n" );writer.write( "}\r\n" );for( ClassPumlVO classPuml:classPumlList ){writer.write( classPuml.toString() );}writer.write( "@enduml\r\n" );} catch (Exception e) {} finally {if (writer != null) {try {writer.close();} catch (Exception e) {}}}}public void generatePuml( Class clazz,String outputPath,boolean ignoreInterface,boolean ignoreProperties ){BufferedWriter writer = null;try {writer = new BufferedWriter(new FileWriter( outputPath ));this.classPumlList = new ArrayList<>();this.generate( clazz,ignoreInterface,ignoreProperties);writer.write( "@startuml\r\n" );writer.write( "!define icon_hammer <img:C:\\E\\sucai\\banshou3.png>\r\n" );writer.write( "!define icon_cube <img:C:\\E\\sucai\\cube_3.png>\r\n" );writer.write( "skinparam Class {\r\n" );writer.write( "\tBackgroundColor  #d3dcef/white\r\n" );writer.write( "}\r\n" );for( ClassPumlVO classPuml:this.classPumlList ){writer.write( classPuml.toString() );}writer.write( "@enduml\r\n" );} catch (Exception e) {} finally {if (writer != null) {try {writer.close();} catch (Exception e) {}}}}private void generate( Class clazz,boolean ignoreInterface,boolean ignoreProperties ){this.generate_inner( clazz,ignoreInterface,ignoreProperties );}public static void main(String[] args) {System.out.println( "xxxx$xxx".contains( "$" ) );}private void generate_inner(Class clazz,boolean ignoreInterface,boolean ignoreProperties) {boolean handleImplementClassList = false;// 只处理 class 和 interfaceif( clazz.isEnum() ){return;}String simpleClassName = clazz.getSimpleName();if( simpleClassName.toLowerCase(  ).endsWith( "properties" ) && ignoreProperties ){return;}// 防止重复处理String classIdentifier = clazz.isInterface() + " " + simpleClassName;if( this.classIdentifiers.contains( classIdentifier ) ){return;}String longClassName = clazz.getName();//  对jdk 以及框架类非业务的class 忽略处理if( longClassName.startsWith( "org." ) ||longClassName.startsWith( "java." ) ||longClassName.startsWith( "sun." ) ||longClassName.startsWith( "com.alibaba.fastjson." ) ||longClassName.startsWith( "tk.mybatis." ) ||longClassName.startsWith( "javax." )){return;}if( JDK_CLASS_NAMES.contains( longClassName ) ){return;}this.classIdentifiers.add( classIdentifier );if( clazz.isInterface() ){if( ignoreInterface ){this.generate_inner_4ImplementClassList( clazz,ignoreInterface,ignoreProperties );return;}else {handleImplementClassList = true;}}ClassPumlVO classPuml = new ClassPumlVO();classPuml.setShortName( simpleClassName );classPuml.setLongName( clazz.getName() );classPuml.setInterface( clazz.isInterface() );this.classPumlList.add( classPuml );//  获取该类直接声明的属性Field[] fields = clazz.getDeclaredFields();if( fields != null && fields.length > 0 ){List<AttributePumlVO> attributePumlList = new ArrayList<>();for( Field field:fields ){String fieldName = field.getName();if( JDK_ATTRIBUTE_NAMES.contains( fieldName ) ){continue;}Class<?> fieldType = field.getType();if( fieldType != null && "org.slf4j.Logger".equals( fieldType.getName() ) ){continue;}AttributePumlVO attributePuml = new AttributePumlVO();attributePuml.setName( fieldName );attributePuml.setType( fieldType );attributePumlList.add( attributePuml );//  对该属性类型对应的 class 进行递归处理this.generate_inner( field.getType(),ignoreInterface,ignoreProperties );}classPuml.setAttributePumlList( attributePumlList );}//  获取该类直接声明的方法Method[] methods = clazz.getDeclaredMethods();if( methods != null && methods.length > 0 ){List<MethodPumlVO> methodPumlList = new ArrayList<>();for( Method method:methods ){String methodName = method.getName();if( JDK_METHOD_NAMES.contains( methodName ) ){continue;}if( methodName.contains( "$" ) ){continue;}MethodPumlVO methodPuml = new MethodPumlVO();methodPuml.setName( methodName );methodPuml.setMethod( method );methodPuml.setReturnType( method.getReturnType() );methodPumlList.add( methodPuml );//  对该方法的返回类型对应的 class 进行递归处理this.generate_inner( method.getReturnType(),ignoreInterface,ignoreProperties );}classPuml.setMethodPumlList( methodPumlList );}if( handleImplementClassList ){// 当前 clazz是接口,获取其全部的实现类,递归调用此方法this.generate_inner_4ImplementClassList(clazz,ignoreInterface,ignoreProperties);}}private void generate_inner_4ImplementClassList(Class clazz, boolean ignoreInterface, boolean ignoreProperties) {if( clazz.getSimpleName().toLowerCase().endsWith( "mapper" ) ){return;}List<Class<?>> implementClassList = this.getImplementClassList4CurrentPackage(clazz);if( implementClassList == null || implementClassList.size() == 0 ){return;}for( Class implementClass:implementClassList ){this.generate_inner( implementClass,ignoreInterface,ignoreProperties );}}private List<Class<?>> getImplementClassList4CurrentPackage(Class clazz){String servicePackage = clazz.getPackage().getName();Reflections reflections = new Reflections(servicePackage);Set<Class<?>> subTypes = reflections.getSubTypesOf( clazz );if( subTypes == null || subTypes.size() == 0 ){return new ArrayList<>( 0 );}return new ArrayList<>(subTypes);}private List<Class<?>> getClasses(String packageName){//第一个class类的集合List<Class<?>> classes = new ArrayList<Class<?>>();//是否循环迭代boolean recursive = true;//获取包的名字 并进行替换String packageDirName = packageName.replace('.', '/');//定义一个枚举的集合 并进行循环来处理这个目录下的thingsEnumeration<URL> dirs;try {dirs = Thread.currentThread().getContextClassLoader().getResources(packageDirName);//循环迭代下去while (dirs.hasMoreElements()){//获取下一个元素URL url = dirs.nextElement();//得到协议的名称String protocol = url.getProtocol();//如果是以文件的形式保存在服务器上if ("file".equals(protocol)) {//获取包的物理路径String filePath = URLDecoder.decode(url.getFile(), "UTF-8");//以文件的方式扫描整个包下的文件 并添加到集合中this.findAndAddClassesInPackageByFile(packageName, filePath, recursive, classes);} else if ("jar".equals(protocol)){//如果是jar包文件//定义一个JarFileJarFile jar;try {//获取jarjar = ((JarURLConnection) url.openConnection()).getJarFile();//从此jar包 得到一个枚举类Enumeration<JarEntry> entries = jar.entries();//同样的进行循环迭代while (entries.hasMoreElements()) {//获取jar里的一个实体 可以是目录 和一些jar包里的其他文件 如META-INF等文件JarEntry entry = entries.nextElement();String name = entry.getName();//如果是以/开头的if (name.charAt(0) == '/') {//获取后面的字符串name = name.substring(1);}//如果前半部分和定义的包名相同if (name.startsWith(packageDirName)) {int idx = name.lastIndexOf('/');//如果以"/"结尾 是一个包if (idx != -1) {//获取包名 把"/"替换成"."packageName = name.substring(0, idx).replace('/', '.');}//如果可以迭代下去 并且是一个包if ((idx != -1) || recursive){//如果是一个.class文件 而且不是目录if (name.endsWith(".class") && !entry.isDirectory()) {//去掉后面的".class" 获取真正的类名String className = name.substring(packageName.length() + 1, name.length() - 6);try {//添加到classesclasses.add(Class.forName(packageName + '.' + className));} catch (ClassNotFoundException e) {}}}}}} catch (IOException e) {}}}} catch (IOException e) {}return classes;}/*** 以文件的形式来获取包下的所有Class* @param packageName* @param packagePath* @param recursive* @param classes*/private void findAndAddClassesInPackageByFile(String packageName, String packagePath, final boolean recursive, List<Class<?>> classes){//获取此包的目录 建立一个FileFile dir = new File(packagePath);//如果不存在或者 也不是目录就直接返回if (!dir.exists() || !dir.isDirectory()) {return;}//如果存在 就获取包下的所有文件 包括目录File[] dirfiles = dir.listFiles(new FileFilter() {//自定义过滤规则 如果可以循环(包含子目录) 或则是以.class结尾的文件(编译好的java类文件)public boolean accept(File file) {return (recursive && file.isDirectory()) || (file.getName().endsWith(".class"));}});//循环所有文件for (File file : dirfiles) {//如果是目录 则继续扫描if (file.isDirectory()) {findAndAddClassesInPackageByFile(packageName + "." + file.getName(),file.getAbsolutePath(),recursive,classes);}else {//如果是java类文件 去掉后面的.class 只留下类名String className = file.getName().substring(0, file.getName().length() - 6);try {//添加到集合中去classes.add(Class.forName(packageName + '.' + className));} catch (ClassNotFoundException e) {}}}}
}

ClassPumlVO.java:
import lombok.Getter;
import lombok.Setter;import java.io.Serializable;
import java.util.List;@Getter
@Setter
public class ClassPumlVO implements Serializable {private boolean isInterface;private String longName;private String shortName;private List<AttributePumlVO> attributePumlList;private List<MethodPumlVO> methodPumlList;@Overridepublic String toString() {StringBuilder sb = new StringBuilder("");if( this.isInterface ){sb.append( "interface" );}else {sb.append( "class" );}sb.append( " " );sb.append( this.shortName );// sb.append( this.longName );sb.append( " {\n" );if( this.attributePumlList != null && this.attributePumlList.size() > 0 ){for( AttributePumlVO attributePuml:this.attributePumlList ){sb.append( attributePuml.toString() );}}if( this.methodPumlList != null && this.methodPumlList.size() > 0 ){for( MethodPumlVO methodPuml:methodPumlList ){sb.append( methodPuml.toString() );}}sb.append( "}\n" );return sb.toString();}
}

MethodPumlVO.java:
import lombok.Getter;
import lombok.Setter;import java.io.Serializable;
import java.lang.reflect.Method;@Getter
@Setter
public class MethodPumlVO implements Serializable {private String name;private Class returnType;private Method method;@Overridepublic String toString() {return "\ticon_cube " + this.name + "(): " + this.returnType.getSimpleName() + "\n";}
}

使用示例:

public static void main(String[] args) throws ClassNotFoundException, IOException {ClassPumlGenerate classPumlGenerate = new ClassPumlGenerate();Class clazz = XxxService.class;String outputPath = "C:\\E\\xxx\\xxx\\xxx\\xxx\\xxx-xxx-xxx\\src\\main\\resources\\puml\\xxx\\puml\\" + clazz.getSimpleName() + ".puml";classPumlGenerate.generatePuml( clazz,outputPath,true,true );}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/120996.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue 使用openlayers导出渲染的地图

下载地图官方示例: http://openlayers.org/en/latest/examples/export-map.html http://openlayers.org/en/latest/examples/export-pdf.html 我这儿使用到了 file-saver (下载图片、附件) npm install file-saver --save import { saveAs } from file-saver; // 执行下…

docker - window Docker Desktop升级

文章目录 前言docker - window Docker Desktop升级 前言 如果您觉得有用的话&#xff0c;记得给博主点个赞&#xff0c;评论&#xff0c;收藏一键三连啊&#xff0c;写作不易啊^ _ ^。   而且听说点赞的人每天的运气都不会太差&#xff0c;实在白嫖的话&#xff0c;那欢迎常来…

【python画画】蘑菇云爱心

来源于网上短视频 数学原理不懂&#xff0c;图个乐 import math from turtle import *def x(i):return 15 * math.sin(i) ** 3 * 20def y(i):return 20 * (12 * math.cos(i) - 5 * math.cos(2 * i) - 2 * math.cos(4 * i))speed(0) color(red) pensize(10) for i in range(51…

【音视频】Linux | FFmpeg源码搭建

Linux | FFmpeg源码搭建 时间:2023-06-21 文章目录 `Linux` | `FFmpeg`源码搭建@[toc]1.参考2.获取源码2-1.建立工作目录2-2.获取`AAC`2-3.获取`X264`2-4.获取`X265`2-5.获取`FFmpeg`3.编译/安装3-1.编译`AAC`3-1-1.解压源码3-1-2.编译3-1-3.安装3-2.编译`X264`3-2-1.解压源码…

Western blot实验步骤

Western blot实验步骤 1 Western blot原理介绍 Western blot是一种通过凝胶电泳分离蛋白质&#xff0c;将其印迹或转移到膜上&#xff0c;并对固定抗原进行选择性免疫检测的技术。这是一种重要的、常规的蛋白质分析方法&#xff0c;依赖于抗体-抗原相互作用的特异性&#xff0…

初级前端面试题(一) 之 html/css/js

目 录 一、 HTML 1. .如何理解HTML语义化的&#xff1f; 2. HTML标签有哪些&#xff1f; 3. Canvas 和SVG的区别 二、CSS 1. BFC是什么&#xff1f; 2. 如何实现垂直居中&#xff1f; 3. css选择器 优先级如何确定&#xff1f; 4. 如何清除浮动&#xff1f; 5. …

Elasticsearch7.8.1集群安装手册

1. 部署说明 elasticsearch集群规划为三个节点&#xff0c;elasticsearch版本为7.8.1 2. 下载安装包 1&#xff09;下载 Elasticsearch7.8.1安装包 # cd /opt # wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-7.8.1-linux-x86_64.tar.gz3. Ela…

C++设计模式_14_Facade门面模式

本篇介绍的Facade门面模式属于“接口隔离”模式的一种&#xff0c;以下进行详细介绍。 文章目录 1. “接口隔离”模式1. 1 典型模式 2. 系统间耦合的复杂度3. 动机(Motivation)4. 模式定义5. Facade门面模式的代码实现6. 结构7. 要点总结8. 其他参考 1. “接口隔离”模式 在组…

笔记44:Batch_Normlization 过程详解

笔记本地地址&#xff1a;D:\work_file\DeepLearning_Learning\03_个人笔记\2.图像处理任务\BN a a a a a a a a a a a a a a a a a

抖音上怎么挂小程序?制作小程序挂载抖音视频

公司企业商家现在已经把抖音作为营销的渠道之一&#xff0c;目前抖音支持短视频挂载小程序&#xff0c;可方便做营销。以下给大家分享这一操作流程。 一、申请自主挂载能力 首先需要在抖音开放平台官网注册一个抖音小程序账号&#xff0c;然后申请短视频自主挂载能力。 二、搭…

Liunx两台服务器实现相互SSH免密登录

一、首先准备两台Linux虚拟机当作此次实验的两台服务器 服务器1&#xff1a;server IPV4&#xff1a;192.168.110.136 服务器2&#xff1a;client IPV4&#xff1a; 192.168.110.134 二、准备阶段 [rootserver ~]# systemctl disable firewalld #关…

Umijs创建一个antd+Ts项目环境

上文搭建Umijs环境并创建一个项目 介绍基本操作中 我们构建了一个Umijs环境的环境 但也只创建了一个页面 真正开发来讲 也不可能只创建几个界面这么简单 这里面的创建 还是非常完整的 这里 我创建一个文件夹 主要是做我们的项目目录 然后 我们在终端输入命令 然后 打开目录终…

C#版字节跳动SDK - SKIT.FlurlHttpClient.ByteDance

前言 在我们日常开发工作中对接第三方开放平台&#xff0c;找一款封装完善且全面的SDK能够大大的简化我们的开发难度和提高工作效率。今天给大家推荐一款C#开源、功能完善的字节跳动SDK&#xff1a;SKIT.FlurlHttpClient.ByteDance。 项目官方介绍 可能是全网唯一的 C# 版字节…

pinia中使用reactive声明变量,子页面使用时,值未改变,即不是响应式的(解决方法)

reactive赋值无效&#xff01;reactive 不要直接data赋值&#xff01;&#xff01;&#xff01;会丢失响应式的&#xff0c;只能通过obj.属性 属性值赋值 方法一. pinia中直接使用ref定义变量即可 export const useUserStoredefineStore(user,()>{let loginUserreactive({…

C语言字符转数字函数

文章目录 atofatoiatolatollstrtodstrtofstrtolstrtoldstrtollstrtoulstrtoull atof double atof (const char* str);Convert string to double Parses the C string str, interpreting its content as a floating point number and returns its value as a double. The func…

【C++进阶】set和map的基本使用(灰常详细)

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前学习C和算法 ✈️专栏&#xff1a;C航路 &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章对你有帮助的话 欢迎 评论&#x1f4ac; 点赞&#x1…

Python 中的内存泄漏问题

内存泄漏是一个常见的编程问题,很难调试和修复。 本文将通过小型和大型示例程序探讨 Python 内存泄漏。 我们将了解如何找到内存泄漏的根源以及如何修复它。 Python 中的内存泄漏 在本文中我们不会讨论 Python 内存管理系统的内部结构。 但是,如果你对Python内存系统是如何…

跳跃游戏 II

给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]。 每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说&#xff0c;如果你在 nums[i] 处&#xff0c;你可以跳转到任意 nums[i j] 处: 0 < j < nums[i] i j < n 返回到达 nums[n - 1] 的最…

基于springboot实现网上图书商城管理系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现网上图书商城管理系统演示 摘要 在Internet高速发展的今天&#xff0c;我们生活的各个领域都涉及到计算机的应用&#xff0c;其中包括网上图书商城的网络应用&#xff0c;在外国网上图书商城已经是很普遍的方式&#xff0c;不过国内的管理网站可能还处于起步…

基于nodejs+vue全国公考岗位及报考人数分析

目 录 摘 要 I ABSTRACT II 目 录 II 第1章 绪论 1 1.1背景及意义 1 1.2 国内外研究概况 1 1.3 研究的内容 1 第2章 相关技术 3 2.1 nodejs简介 4 2.2 express框架介绍 6 2.4 MySQL数据库 4 第3章 系统分析 5 3.1 需求分析 5 3.2 系统可行性分析 5 3.2.1技术可行性&#xff1a;…