我正在尝试制作一个 Java 工具来扫描 Java 应用程序的结构并提供一些有意义的信息。为此,我需要能够从项目位置(JAR/WAR 或只是一个文件夹)扫描所有 .class 文件,并使用反射来了解它们的方法。事实证明这几乎是不可能的。
我可以找到很多基于 URLClassloader 的解决方案,这些解决方案允许我从目录/存档加载特定类,但没有一个解决方案允许我在没有任何关于类名或包结构的信息的情况下加载类。
编辑: 我觉得我表达得很糟糕。我的问题不是我无法获得所有的类文件,我可以通过递归等来做到这一点并正确定位它们。我的问题是为每个类文件获取一个类对象。
最佳答案
以下代码从 JAR 文件加载所有类。它不需要知道任何关于类的信息。类的名称是从 JarEntry 中提取的。
JarFile jarFile = new JarFile(pathToJar);
Enumeration<JarEntry> e = jarFile.entries();
URL[] urls = { new URL("jar:file:" + pathToJar+"!/") };
URLClassLoader cl = URLClassLoader.newInstance(urls);
while (e.hasMoreElements()) {
JarEntry je = e.nextElement();
if(je.isDirectory() || !je.getName().endsWith(".class")){
continue;
}
// -6 because of .class
String className = je.getName().substring(0,je.getName().length()-6);
className = className.replace('/', '.');
Class c = cl.loadClass(className);
}
编辑:
正如上面评论中所建议的,javassist 也是一种可能性。 在上面代码的 while 循环之前的某个地方初始化一个 ClassPool,而不是使用类加载器加载类,您可以创建一个 CtClass 对象:
ClassPool cp = ClassPool.getDefault();
...
CtClass ctClass = cp.get(className);
从 ctClass 中,您可以获取所有方法、字段、嵌套类...... 看一下 javassist api: https://jboss-javassist.github.io/javassist/html/index.html
关于java - 如何在运行时从文件夹或 JAR 加载类?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11016092/