【发布时间】:2011-02-02 15:39:26
【问题描述】:
我想获得属于某个包的所有类及其所有子类的列表。这些类可能已加载到 JVM 中,也可能尚未加载。
【问题讨论】:
标签: java reflection jvm
我想获得属于某个包的所有类及其所有子类的列表。这些类可能已加载到 JVM 中,也可能尚未加载。
【问题讨论】:
标签: java reflection jvm
这不是程序化解决方案,但您可以运行
java -verbose:class ....
JVM 会转储它正在加载的内容,以及从哪里加载。
[Opened /usr/java/j2sdk1.4.1/jre/lib/rt.jar]
[Opened /usr/java/j2sdk1.4.1/jre/lib/sunrsasign.jar]
[Opened /usr/java/j2sdk1.4.1/jre/lib/jsse.jar]
[Opened /usr/java/j2sdk1.4.1/jre/lib/jce.jar]
[Opened /usr/java/j2sdk1.4.1/jre/lib/charsets.jar]
[Loaded java.lang.Object from /usr/java/j2sdk1.4.1/jre/lib/rt.jar]
[Loaded java.io.Serializable from /usr/java/j2sdk1.4.1/jre/lib/rt.jar]
[Loaded java.lang.Comparable from /usr/java/j2sdk1.4.1/jre/lib/rt.jar]
[Loaded java.lang.CharSequence from /usr/java/j2sdk1.4.1/jre/lib/rt.jar]
[Loaded java.lang.String from /usr/java/j2sdk1.4.1/jre/lib/rt.jar]
更多详情请见here。
【讨论】:
使用Reflections 库,很简单:
Reflections reflections = new Reflections("my.pkg", new SubTypesScanner(false));
这将扫描包含 my.pkg 包的 url/s 中的所有类。
因此,获取所有类实际上是获取 Object 的所有子类型,可传递:
Set<String> allClasses =
reflections.getStore().getSubTypesOf(Object.class.getName());
(reflections.getSubTypesOf(Object.class) 的普通方式会导致将 all 类加载到 PermGen 中,并且可能会抛出 OutOfMemoryError。你不想这样做......)
如果您想获得 Object(或任何其他类型)的所有 直接 子类型,而不是一次性获得其传递子类型,请使用以下命令:
Collection<String> directSubtypes =
reflections.getStore().get(SubTypesScanner.class).get(Object.class.getName());
【讨论】:
这个问题有多个答案,部分原因是问题含糊不清 - 标题是关于 JVM 加载的类,而问题的内容是“可能由 JVM 加载,也可能不被 JVM 加载”。
假设 OP 需要通过给定类加载器由 JVM 加载的类,并且只有那些类 - 我也需要 - 有一个解决方案 (elaborated here),如下所示:
import java.net.URL;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.Vector;
public class CPTest {
private static Iterator list(ClassLoader CL)
throws NoSuchFieldException, SecurityException,
IllegalArgumentException, IllegalAccessException {
Class CL_class = CL.getClass();
while (CL_class != java.lang.ClassLoader.class) {
CL_class = CL_class.getSuperclass();
}
java.lang.reflect.Field ClassLoader_classes_field = CL_class
.getDeclaredField("classes");
ClassLoader_classes_field.setAccessible(true);
Vector classes = (Vector) ClassLoader_classes_field.get(CL);
return classes.iterator();
}
public static void main(String args[]) throws Exception {
ClassLoader myCL = Thread.currentThread().getContextClassLoader();
while (myCL != null) {
System.out.println("ClassLoader: " + myCL);
for (Iterator iter = list(myCL); iter.hasNext();) {
System.out.println("\t" + iter.next());
}
myCL = myCL.getParent();
}
}
}
其中一个巧妙之处在于您可以选择要检查的任意类加载器。然而,如果类加载器类的内部发生变化,它可能会中断,因此它被用作一次性诊断工具。
【讨论】:
Thread.currentThread().getContextClassLoader() 与CPTest.class.getClassLoader() 有什么区别?您是否有理由更喜欢从 Thread 对象中获取 ClassLoader?
ClassLoader.class.getDeclaredField("classes")来避免while循环。
上述方法的另一种方法是使用java.lang.instrument 创建一个外部代理,以找出加载了哪些类并使用-javaagent 开关运行您的程序:
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.security.ProtectionDomain;
public class SimpleTransformer implements ClassFileTransformer {
public SimpleTransformer() {
super();
}
public byte[] transform(ClassLoader loader, String className, Class redefiningClass, ProtectionDomain domain, byte[] bytes) throws IllegalClassFormatException {
System.out.println("Loading class: " + className);
return bytes;
}
}
这种方法的另一个好处是为您提供有关哪个 ClassLoader 加载了给定类的信息。
【讨论】:
我还建议您编写一个-javagent 代理,但使用getAllLoadedClasses 方法而不是转换任何类。
要与您的客户端代码(普通 Java 代码)同步,请创建一个套接字并通过它与代理通信。然后,您可以在需要时触发“列出所有类”方法。
【讨论】:
如果您已经知道包顶级路径,一种方法是使用OpenPojo
final List<PojoClass> pojoClasses = PojoClassFactory.getPojoClassesRecursively("my.package.path", null);
然后您可以浏览列表并执行您想要的任何功能。
【讨论】:
该程序将打印所有类及其物理路径。如果您需要分析从任何 Web/应用程序服务器加载的类,use 可以简单地将其复制到任何 JSP。
import java.lang.reflect.Field;
import java.util.Vector;
public class TestMain {
public static void main(String[] args) {
Field f;
try {
f = ClassLoader.class.getDeclaredField("classes");
f.setAccessible(true);
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
Vector<Class> classes = (Vector<Class>) f.get(classLoader);
for(Class cls : classes){
java.net.URL location = cls.getResource('/' + cls.getName().replace('.',
'/') + ".class");
System.out.println("<p>"+location +"<p/>");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
【讨论】:
【讨论】:
您可能能够获得通过类加载器加载的类的列表,但这不包括您尚未加载但在类路径中的类。
要获得类路径中的所有类,您必须执行类似于第二个解决方案的操作。如果您确实想要当前“已加载”的类(换句话说,您已经引用、访问或实例化的类),那么您应该细化您的问题以表明这一点。
【讨论】:
在 JRockit JVM 下运行您的代码,然后使用 JRCMD <PID> print_class_summary
这将输出所有加载的类,每行一个。
【讨论】:
好吧,我所做的只是列出类路径中的所有文件。这可能不是一个光荣的解决方案,但它可靠地工作,并为我提供了我想要的一切,等等。
【讨论】: