作者:小丁啊小丁 | 来源:互联网 | 2023-09-18 20:06
我已经搜索了一段时间,似乎有不同的方法,这里是一个摘要:
如果你不介意添加依赖项,则反射库非常受欢迎。它看起来像这样:
Reflections reflectiOns= new Reflections("firstdeveloper.examples.reflections");
Set> classes = reflections.getSubTypesOf(Pet.class);
ServiceLoader(根据埃里克森的回答),看起来像这样:
ServiceLoader loader = ServiceLoader.load(Pet.class);
for (Pet implClass : loader) {
System.out.println(implClass.getclass().getSimpleName()); // prints Dog, Cat
}
请注意,要使其正常工作,你需要定义Pet
为ServiceProviderInterface(SPI)
并声明其实现。你这样做,通过创建一个文件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
的方法,这些方法将始终限于已加载的类,除非你执行基于目录的搜索。