前段时间,我遇到了一段代码,它使用了一些标准的Java功能来定位实现给定接口的类.我知道函数隐藏在某些非逻辑位置,但它们可以用于其他类,因为包含的名称是隐含的.那时我不需要它,所以我忘了它,但现在我做了,我似乎无法再找到这些功能.这些功能在哪里可以找到?
编辑:我不是在寻找任何IDE函数或任何东西,而是可以在Java应用程序中执行的东西.
Bri*_*per 63
不久之前,我整理了一个包,用于做你想做的事情,等等.(我需要它用于我正在编写的实用程序).它使用ASM库.您可以使用反射,但ASM表现更好.
我将我的包放在我网站上的开源库中.该库位于:http://software.clapper.org/javautil/.您想从使用ClassFinder类开始.
我为它编写的实用程序是一个RSS阅读器,我仍然每天使用它,所以代码确实倾向于运用.我使用ClassFinder来支持RSS阅读器中的插件API; 在启动时,它在几个目录树中查找包含实现特定接口的类的jar和类文件.它比你想象的要快得多.
该库是BSD许可的,因此您可以安全地将其与您的代码捆绑在一起.来源可用.
如果这对你有用,请自己帮忙.
更新:如果您正在使用Scala,您可能会发现此库对Scala更友好.
小智 51
春天可以为你做到这一点......
BeanDefinitionRegistry bdr = new SimpleBeanDefinitionRegistry();
ClassPathBeanDefinitionScanner s = new ClassPathBeanDefinitionScanner(bdr);
TypeFilter tf = new AssignableTypeFilter(CLASS_YOU_WANT.class);
s.addIncludeFilter(tf);
s.scan("package.you.want1", "package.you.want2");
String[] beans = bdr.getBeanDefinitionNames();
Run Code Online (Sandbox Code Playgroud)
注意如果您想要正确的结果,TypeFilter很重要!您也可以在此处使用排除过滤器.
Scanner可以在spring-context jar中找到,spring-beans中的注册表,类型过滤器是spring-core.
eri*_*son 22
你正在谈论的代码听起来ServiceLoader,这是在Java 6中推出了支持,因为已定义功能的Java 1.3或更早版本.出于性能原因,这是在运行时查找接口实现的推荐方法; 如果您需要在旧版本的Java中支持此功能,我希望您会发现我的实现很有帮助.
在早期版本的Java中有几个这样的实现,但是在Sun包中,而不是在核心API中(我认为ImageIO内部有一些类来执行此操作).由于代码很简单,我建议您提供自己的实现,而不是依赖于可能会发生变化的非标准Sun代码.
Ada*_*ent 13
包级别注释
我知道很久以前这个问题已经得到解答了,但是这个问题的另一个解决方案是使用Package Level Annotations.
虽然很难找到JVM中的所有类,但实际上很容易浏览包层次结构.
Package[] ps = Package.getPackages();
for (Package p : ps) {
MyAno a = p.getAnnotation(MyAno.class)
// Recursively descend
}
Run Code Online (Sandbox Code Playgroud)
然后让你的注释有一个Class数组的参数.然后在package-info.java中为特定包放置MyAno.
如果人们有兴趣,我会添加更多细节(代码),但最有可能得到这个想法.
MetaInf服务加载器
要添加到@erickson答案,您还可以使用服务加载器方法.Kohsuke有一种很棒的方法来生成服务加载器方法所需的META-INF内容:
http://weblogs.java.net/blog/kohsuke/archive/2009/03/my_project_of_t.html
小智 9
您还可以使用可扩展组件扫描程序(extcos:http://sf.net/projects/extcos )并搜索所有实现如下界面的类:
Set<Class<? extends MyInterface>> classes = new HashSet<Class<? extends MyInterface>>();
ComponentScanner scanner = new ComponentScanner();
scanner.getClasses(new ComponentQuery() {
@Override
protected void query() {
select().
from("my.package1", "my.package2").
andStore(thoseImplementing(MyInterface.class).into(classes)).
returning(none());
}
});
Run Code Online (Sandbox Code Playgroud)
这适用于文件系统上的类,在jar中,甚至适用于JBoss虚拟文件系统上的类.它进一步设计用于独立应用程序以及任何Web或应用程序容器.
完全一般,这个功能是不可能的.Java ClassLoader机制仅保证能够请求具有特定名称的类(包括pacakge),并且ClassLoader可以提供类,或者它可以声明它不知道该类.
可以(并且经常)从远程服务器加载类,甚至可以在运行中构建它们; 编写一个ClassLoader并不难,它返回一个有效的类,为你从中提出的任何名称实现给定的接口; 实现该接口的类的列表的长度将是无限的.
实际上,最常见的情况是URLClassLoader在文件系统目录和JAR文件列表中查找类.因此,您需要获取URLClassLoader,然后遍历这些目录和存档,并为您在其中找到的每个类文件,请求相应的Class对象并查看其getInterfaces()方法的返回.
显然,Class.isAssignableFrom()会告诉您单个类是否实现给定的接口.那么问题是获取要测试的类列表.
据我所知,Java没有直接的方式向类加载器询问" 可能加载的类列表".所以你必须通过遍历可见的jar来自己做,调用Class.forName()来加载类,然后测试它.
但是,如果您只想知道实际加载的那些实现给定接口的类,那么它会更容易一些:
如果您使用检测技术,那么(如链接中所述)会发生什么是在JVM启动时调用"代理"类并传递Instrumentation对象.此时,您可能希望在静态字段中"将其保存以供日后使用",然后让主应用程序代码稍后调用它以获取已加载类的列表.
| 归档时间: |
|
| 查看次数: |
166768 次 |
| 最近记录: |