如何在Java中以编程方式获取接口的所有实现的列表?

2022-08-31 10:38:43

我可以用反射或类似的东西来做吗?


答案 1

我一直在寻找一段时间,似乎有不同的方法,这里有一个总结:

  1. 反射库非常受欢迎,如果你不介意添加依赖关系。它看起来像这样:

    Reflections reflections = new Reflections("firstdeveloper.examples.reflections");
    Set<Class<? extends Pet>> classes = reflections.getSubTypesOf(Pet.class);
    
  2. ServiceLoader(根据埃里克森的答案),它看起来像这样:

    ServiceLoader<Pet> loader = ServiceLoader.load(Pet.class);
    for (Pet implClass : loader) {
        System.out.println(implClass.getClass().getSimpleName()); // prints Dog, Cat
    }
    

    请注意,要实现这一点,您需要定义为服务提供程序接口 (SPI) 并声明其实现。您可以通过在 中创建一个带有名称的文件并在其中声明 的所有实现来做到这一点Petresources/META-INF/servicesexamples.reflections.PetPet

    examples.reflections.Dog
    examples.reflections.Cat
    
  3. 包级批注。下面是一个示例:

    Package[] packages = Package.getPackages();
    for (Package p : packages) {
        MyPackageAnnotation annotation = p.getAnnotation(MyPackageAnnotation.class);
        if (annotation != null) {
            Class<?>[]  implementations = annotation.implementationsOfPet();
            for (Class<?> impl : implementations) {
                System.out.println(impl.getSimpleName());
            }
        }
    }
    

    和注释定义:

    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.PACKAGE)
    public @interface MyPackageAnnotation {
        Class<?>[] implementationsOfPet() default {};
    }
    

    并且必须在该包内命名的文件中声明包级批注。以下是示例内容:package-info.java

    @MyPackageAnnotation(implementationsOfPet = {Dog.class, Cat.class})
    package examples.reflections;
    

    请注意,只有 ClassLoader 当时已知的包才会通过调用 来加载。Package.getPackages()

此外,还有其他基于 URLClassLoader 的方法,这些方法将始终限制为已加载的类,除非您执行基于目录的搜索。


答案 2

埃里克森说了什么,但如果你还想这样做,那就看看Reflections。从他们的页面:

使用反射,您可以查询元数据:

  • 获取某种类型的所有子类型
  • 获取带有一些注释的所有类型
  • 获取使用某些注释注释的所有类型,包括匹配的注释参数
  • 获取所有方法,并用一些方法注释