纯原生Java实现:获取整个项目中指定接口所有的实现类

不使用第三方,不使用属性文件,不指定包名,获取整个系统中某一个接口所有的实现类,纯Java实现

java 复制代码
/**
 * 类查找器,用于扫描类路径中的所有类,并找出指定类的实现类。
 * 该类通过递归扫描类路径下的所有 .class 文件,加载并判断是否为目标类的实现类。
 */
public class ClassFinder {
    /**
     * 获取指定类的所有实现类(非接口本身)。
     * 从类路径根目录开始扫描,使用当前线程的类加载器获取资源路径。
     *
     * @param clazz 要查找实现类的目标类(通常是抽象类或接口)
     * @return 返回目标类的所有实现类组成的列表
     */
    public static List<Class<?>> getImplementationsOfMyService(Class<?> clazz) {
        List<Class<?>> implementations = new ArrayList<>();
        // 设置为空字符串表示从类路径根开始扫描
        String packageName = "";

        // 使用当前线程的类加载器加载类资源
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        Enumeration<URL> resources = null;
        try {
            // 获取类路径下对应包名的资源路径(转换为文件系统路径格式)
            resources = classLoader.getResources(packageName.replace('.', '/'));
            while (resources.hasMoreElements()) {
                URL resource = resources.nextElement();
                File file = new File(resource.getFile());
                // 开始递归扫描该目录下的类文件
                scanDirectory(clazz, file, "", implementations);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        return implementations;
    }

    /**
     * 递归扫描目录及其子目录中的类文件。
     * 对每个类文件进行加载并判断是否是目标类的实现类。
     *
     * @param clazz 目标类(要查找其实现类)
     * @param dir 当前扫描的目录
     * @param pkg 当前目录对应的包名前缀
     * @param list 存储符合条件的实现类
     */
    private static void scanDirectory(Class<?> clazz, File dir, String pkg, List<Class<?>> list) {
        if (!dir.exists()) return;

        for (File file : Objects.requireNonNull(dir.listFiles())) {
            if (file.isDirectory()) {
                // 如果是目录,递归进入子目录继续扫描
                scanDirectory(clazz, file, pkg + file.getName() + ".", list);
            } else if (file.getName().endsWith(".class")) {
                // 构建完整的类名(包含包名)
                String className = pkg + file.getName().replace(".class", "");
                try {
                    // 加载类
                    Class<?> tmp = Class.forName(className);
                    // 判断是否为目标类的实现类,并排除接口类型
                    if (clazz.isAssignableFrom(tmp) && !tmp.isInterface()) {
                        list.add(tmp);
                    }
                } catch (Exception ignored) {
                    // 忽略加载失败的类
                }
            }
        }
    }

    // 测试入口:演示如何使用ClassFinder查找AnnotationHandler接口的实现类
    public static void main(String[] args) throws Exception {
        // 查找AnnotationHandler接口的所有实现类
        List<Class<?>> classes = getImplementationsOfMyService(AnnotationHandler.class);
        System.out.println("找到以下实现类:");
        for (Class<?> c : classes) {
            System.out.println(c.getName());
        }
    }
}
相关推荐
zc.z13 小时前
JAVA实现:纯PCM格式音频转换成BASE64
java·音视频·pcm
mask哥13 小时前
力扣算法java实现汇总整理(上)
java·算法·leetcode
Aaswk15 小时前
Java Lambda 表达式与流处理
java·开发语言·python
是宇写的啊15 小时前
Spring AOP
java·spring
万邦科技Lafite15 小时前
京东item_get接口实战案例:实时商品价格监控全流程解析
java·开发语言·数据库·python·开放api·淘宝开放平台
Mr_pyx16 小时前
Spring AI 入门教程:Java开发者的AI应用捷径
java·人工智能·spring
Zephyr_017 小时前
Leedcode算法题
java·算法
苍煜17 小时前
Java开发IO零基础吃透:BIO、NIO、同步异步、阻塞非阻塞
java·python·nio
折哥的程序人生 · 物流技术专研18 小时前
Java面试85题图解版(一):基础核心篇
java·开发语言·后端·面试
AllData公司负责人18 小时前
通过Postgresql同步到Doris,全视角演示AllData数据中台核心功能效果,涵盖:数据入湖仓,数据同步,数据处理,数据服务,BI可视化驾驶舱
java·大数据·数据库·数据仓库·人工智能·python·postgresql