【工具】将类里的属性解析成属性描述的JSONArray数组

新建一个类,将类里的属性逐个解析出来,用一个json对象封装每一个属性的描述,展示上下级关系;

maven依赖
复制代码
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.83</version>
        </dependency>

        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-all</artifactId>
            <version>4.1.19</version>
        </dependency>
PropertyTypesUtil
复制代码
package com.xx.util;

import cn.hutool.core.util.ReflectUtil;
import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;

import java.lang.reflect.Field;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Objects;

public class PropertyTypesUtil {

    // 白名单包名
    final static String PKG_NAME = "com.xx";

    /**
     * 解析类的参数列表
     *
     * @param clazz
     * @return
     */
    public static JSONArray parsePropertyTypes(Class clazz) {
        JSONArray allInput = new JSONArray();
        getPropertyTypes(clazz, allInput, null, null);
        return allInput;
    }

    /**
     * 递归获取类的属性类型
     */
    public static void getPropertyTypes(Class<?> clazz, JSONArray allInput, String parentFieldName, String parentPath) {
        // 获取当前类的所有属性
        Field[] fields = ReflectUtil.getFields(clazz);
        for (Field field : fields) {
            // 获取属性名称和类型
            String propertyName = field.getName();
            Class<?> propertyType = field.getType();
            String path = (Objects.isNull(parentPath) ? "" : parentPath + ".") + propertyName;

            if (!Objects.equals("serialVersionUID", propertyName)) {
                // 存储属性类型信息
                JSONObject obj = new JSONObject();
                obj.put("code", propertyName);
                obj.put("parentCode", parentFieldName);
                obj.put("type", propertyType.getName().startsWith(PKG_NAME) ? "object" : propertyType.getSimpleName());
                obj.put("path", path);
                allInput.add(obj);
            }

            // 如果是集合,还需要把泛型解析出来
            if (List.class.isAssignableFrom(propertyType)) {
                Type genericType = field.getGenericType();
                if (genericType instanceof ParameterizedType) {
                    ParameterizedType pt = (ParameterizedType) genericType;
                    Type[] actualTypeArguments = pt.getActualTypeArguments();
                    if (actualTypeArguments.length > 0) {
                        Type actualTypeArgument = actualTypeArguments[0];
                        try {
                            getPropertyTypes(Class.forName(actualTypeArguments[0].getTypeName()), allInput, propertyName, path + "[]");
                        } catch (Exception e) {
                            //log.error("解析:{}异常", actualTypeArgument.toString());
                        }
                    }
                }
            }

            // 如果属性类型是一个类(而不是基本类型),则递归获取这个类的属性类型
            if (!propertyType.isPrimitive() && propertyType.getName().startsWith(PKG_NAME)) {
                getPropertyTypes(propertyType, allInput, propertyName, path);
            }
        }

        // 获取父类的属性类型
        if (clazz.getSuperclass() != null && !clazz.getSuperclass().equals(Object.class)) {
            getPropertyTypes(clazz.getSuperclass(), allInput, null, null);
        }
    }
}
新增两个类

Pojo

复制代码
package com.xx.util;

import lombok.Getter;
import lombok.Setter;

@Getter
@Setter
public class Pojo {
    private String name;
    private String address;

    private SubPojo subPojo;
}

SubPojo

复制代码
package com.xx.util;

import lombok.Getter;
import lombok.Setter;

@Getter
@Setter
public class SubPojo {
    private String email;
    private String qq;
}
测试示例
复制代码
    @Test
    public void runTest() {
        JSONArray objects = PropertyTypesUtil.parsePropertyTypes(Pojo.class);
        System.out.println(objects);
    }
测试结果
复制代码
[
    {
        "path": "name",
        "code": "name",
        "type": "String"
    },
    {
        "path": "address",
        "code": "address",
        "type": "String"
    },
    {
        "path": "subPojo",
        "code": "subPojo",
        "type": "object"
    },
    {
        "path": "subPojo.email",
        "code": "email",
        "parentCode": "subPojo",
        "type": "String"
    },
    {
        "path": "subPojo.qq",
        "code": "qq",
        "parentCode": "subPojo",
        "type": "String"
    }
]

over~~

相关推荐
二哈赛车手8 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
栗子~~9 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8299 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
未若君雅裁10 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
AI人工智能+电脑小能手10 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
阿维的博客日记11 小时前
Nacos 为什么能让配置动态生效?(涉及 @RefreshScope 注解)
java·spring
雨辰AI11 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务
辰海Coding12 小时前
MiniSpring框架学习-完成的 IoC 容器
java·spring boot·学习·架构
小小编程路12 小时前
C++ 多线程与并发
java·jvm·c++
AI视觉网奇12 小时前
linux 检索库 判断库是否支持
java·linux·服务器