温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

Java中怎么利用反射获取某个接口

发布时间:2021-06-18 15:16:31 来源:亿速云 阅读:409 作者:Leah 栏目:大数据

Java中怎么利用反射获取某个接口,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

/**

 * 获取一个接口的所有实现类
 *
 * @param target
 * @return
 */
public static ArrayList<Class<?>> getInterfaceImpls(Class<?> target) {
    ArrayList<Class<?>> subclassaes = Lists.newArrayList();
    try {
        // 判断class对象是否是一个接口
        if (target.isInterface()) {
            @NotNull
            String basePackage = target.getClassLoader().getResource("").getPath();
            File[] files = new File(basePackage).listFiles();
            // 存放class路径的list
            ArrayList<String> classpaths = Lists.newArrayList();
            for (File file : files) {
                // 扫描项目编译后的所有类
                if (file.isDirectory()) {
                    listPackages(file.getName(), classpaths);
                }
            }
            // 获取所有类,然后判断是否是 target 接口的实现类
            for (String classpath : classpaths) {
                Class<?> classObject = Class.forName(classpath);
                if (classObject.getSuperclass() == null) continue; // 判断该对象的父类是否为null
                Set<Class<?>> interfaces = new HashSet<>(Arrays.asList(classObject.getInterfaces()));
                if (interfaces.contains(target)) {
                    subclasses.add(Class.forName(classObject.getName()));
                }
            }
        } else {
            throw new ParamException("Class对象不是一个interface");
        }
    } catch (Throwable e) {
        e.printStackTrace();
    }
    return subclasses;
}

/**
 * 获取项目编译后的所有的.class的字节码文件
 * 这么做的目的是为了让 Class.forName() 可以加载类
 *
 * @param basePackage 默认包名
 * @param classes     存放字节码文件路径的集合
 * @return
 */
public static void listPackages(String basePackage, List<String> classes) {
    URL url = SophonUtils.class.getClassLoader()
            .getResource("./" + basePackage.replaceAll("\\.", "/"));
    File directory = new File(url.getFile());
    for (File file : directory.listFiles()) {
        // 如果是一个目录就继续往下读取(递归调用)
        if (file.isDirectory()) {
            listPackages(basePackage + "." + file.getName(), classes);
        } else {
            // 如果不是一个目录,判断是不是以.class结尾的文件,如果不是则不作处理
            String classpath = file.getName();
            if (".class".equals(classpath.substring(classpath.length() - ".class".length()))) {
                classes.add(basePackage + "." + classpath.replaceAll(".class", ""));
            }
        }
    }
}

代码演示:

使用方法非常简单,你只需要调用getInterfaceImpls()方法即可,listPackages()方法是个辅助。

//
// getInstanceImpls()返回一个Class<?>对象数组
// 这个数组中包含的数据就是SophonInit接口的子类
//
ArrayList<Class<?>> subclass = getInterfaceImpls(SophonInit.class);

ps:值的注意的地方是,这个方法只能获取项目中自己定义的接口,不能获取到JDK或者是其他Jar包中的接口,因为这个工具的原理就是扫描编译后的classes目录

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注亿速云行业资讯频道,感谢您对亿速云的支持。

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI