如何在Java中以编程方式获取接口的所有实现的列表?
我可以用反射或类似的东西来做吗?
我一直在寻找一段时间,似乎有不同的方法,这里有一个总结:
反射库非常受欢迎,如果你不介意添加依赖关系。它看起来像这样:
Reflections reflections = new Reflections("firstdeveloper.examples.reflections");
Set<Class<? extends Pet>> classes = reflections.getSubTypesOf(Pet.class);
ServiceLoader(根据埃里克森的答案),它看起来像这样:
ServiceLoader<Pet> loader = ServiceLoader.load(Pet.class);
for (Pet implClass : loader) {
System.out.println(implClass.getClass().getSimpleName()); // prints Dog, Cat
}
请注意,要实现这一点,您需要定义为服务提供程序接口 (SPI) 并声明其实现。您可以通过在 中创建一个带有名称的文件并在其中声明 的所有实现来做到这一点Pet
resources/META-INF/services
examples.reflections.Pet
Pet
examples.reflections.Dog
examples.reflections.Cat
包级批注。下面是一个示例:
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 的方法,这些方法将始终限制为已加载的类,除非您执行基于目录的搜索。
埃里克森说了什么,但如果你还想这样做,那就看看Reflections。从他们的页面:
使用反射,您可以查询元数据:
- 获取某种类型的所有子类型
- 获取带有一些注释的所有类型
- 获取使用某些注释注释的所有类型,包括匹配的注释参数
- 获取所有方法,并用一些方法注释