通过包名和注解获取类Class

import java.io.File;
import java.io.IOException;
import java.lang.annotation.Annotation;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
// 采用ClassLoader扫描
public class ClassUtils{
    /**
     * 从包package中获取所有的Class
     *
     * @param packageName
     * @return
     */
    public Set<Class<?>> getClasses(String packageName) throws Exception {

        // 第一个class类的集合
        //List<Class<?>> classes = new ArrayList<Class<?>>();
        Set<Class<?>> classes = new HashSet<>();
        // 是否循环迭代
        boolean recursive = true;
        // 获取包的名字 并进行替换
        String packageDirName = packageName.replace('.', '/');
        // 定义一个枚举的集合 并进行循环来处理这个目录下的things
        Enumeration<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");
                    // 以文件的方式扫描整个包下的文件 并添加到集合中
                    addClass(classes, filePath, packageName);
                } else if ("jar".equals(protocol)) {
                    // 如果是jar包文件
                    // 定义一个JarFile
                    JarFile jar;
                    try {
                        // 获取jar
                        jar = ((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 {
                                            // 添加到classes
                                            classes.add(Class.forName(packageName + '.' + className));
                                        } catch (ClassNotFoundException e) {
                                            e.printStackTrace();
                                        }
                                    }
                                }
                            }
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }

        return classes;
    }

    public void addClass(Set<Class<?>> classes, String filePath, String packageName) throws Exception {
        File[] files = new File(filePath).listFiles(file -> (file.isFile() && file.getName().endsWith(".class")) || file.isDirectory());
        assert files != null;
        for (File file : files) {
            String fileName = file.getName();
            if (file.isFile()) {
                String classsName = fileName.substring(0, fileName.lastIndexOf("."));
                if (!packageName.isEmpty()) {
                    classsName = packageName + "." + classsName;
                }
                doAddClass(classes, classsName);
            }

        }
    }

    public void doAddClass(Set<Class<?>> classes, final String classsName) throws Exception {
        ClassLoader classLoader = new ClassLoader() {
            @Override
            public Class<?> loadClass(String name) throws ClassNotFoundException {
                return super.loadClass(name);
            }
        };
        classes.add(classLoader.loadClass(classsName));
    }


    public <A extends Annotation> Set<Class<?>> getAnnotationClasses(String packageName, Class<A> annotationClass) throws Exception {

        //找用了annotationClass注解的类
        Set<Class<?>> controllers = new HashSet<>();
        Set<Class<?>> clsList = getClasses(packageName);
        if (clsList != null && clsList.size() > 0) {
            for (Class<?> cls : clsList) {
                if (cls.getAnnotation(annotationClass) != null) {
                    controllers.add(cls);
                }
            }
        }
        return controllers;
    }


}

自定义返回数据到前端

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.tecloman.web.common.annotation.LangEntity;
import com.tecloman.web.common.annotation.Scanner;
import com.tecloman.web.common.utils.LangUtils;

import java.io.IOException;
import java.lang.reflect.Field;
import java.util.Set;

/**
 * 中英文转换
 * 2022-04-07 chens
 */
public class LangShift extends JsonSerializer<Object> {
    // 获取LangEntity下的所有类名
    public static void main(String[] args) throws Exception {
        Set<Class<?>> set = new ClassUtils().getAnnotationClasses("com.tecloman.web.modules.hss.entity", LangEntity.class);
        for (Class<?> aClass : set) {
            Field[] fields = aClass.getDeclaredFields();
            System.out.println(fields.length);
            for (Field field : fields) {
                System.out.println(field.getGenericType().toString());
                //System.out.println(field);
            }
            System.out.println(aClass.getName());
        }
    }
    // 也可以通过下面这种方式获取
    // 获取打了Lang注解的类class
    public static void main(String[] args) throws Exception {
        Set<Class<?>> set = new Scanner().getAnnotationClasses("com.tecloman.web.modules.hss.entity", LangEntity.class);
        for (Class<?> aClass : set) {
            Field[] fields = aClass.getDeclaredFields();
            System.out.println(fields.length);
            for (Field field : fields) {
                System.out.println(field.getGenericType().toString());
                // 判断字段上是否存在注解
                boolean present = field.isAnnotationPresent(Lang.class);
                if (present){
                    // 获取注解上的值
                    System.out.println(field.getAnnotation(Lang.class).value());
                }
            }
            System.out.println(aClass.getName());
        }
    }
    // 中英文转换  LangUtils.translate方法会将中文转成英文返回前端
    // 上面的代码只是测试获取指定注解的类,和下面代码无关
    @Override
    public void serialize(Object value, JsonGenerator gen, SerializerProvider serializers) throws IOException {
        //返回数据给前端 value是属性值 调用实体类get方法获取,可以改变赋值
         gen.writeString(LangUtils.translate(value.toString()));
    }
}

实体类打上@JsonSerialize注解
有一些表的主键是雪花生成的比较长,返回前端会出现精度损失,转成字符串返回就可以了
@JsonSerialize(using = ToStringSerializer.class)

@Data
@LangEntity
public class HssEquipmentEntity implements Serializable {
    private static final long serialVersionUID = 1L;

    @TableId
    //  @NotNull(message = "ID不能为空", groups = UpdateGroup.class) // 修改才会触发
    private Long id;
    // @Null(message = "项目ID不能修改", groups = UpdateGroup.class)
    private Long projectId;
    // Long返回前端会出现四舍五入的情况
    @JsonSerialize(using = ToStringSerializer.class)
    private Long orderId;
    private Long mqttId;
    /**
     * 名称
     */
    @JsonSerialize(using = LangShift.class)
    private String name;
    @JsonSerialize(using = LangShift.class)
    private String sn;

也可以直接在实体的get方法上更改数据,但效率慢