【问题标题】:Java - Get a list of all Classes loaded in the JVMJava - 获取 JVM 中加载的所有类的列表
【发布时间】:2011-02-02 15:39:26
【问题描述】:

我想获得属于某个包的所有类及其所有子类的列表。这些类可能已加载到 JVM 中,也可能尚未加载。

【问题讨论】:

    标签: java reflection jvm


    【解决方案1】:

    这不是程序化解决方案,但您可以运行

    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

    【讨论】:

    • 问题询问“获取 JVM 中加载的所有类的列表”,而不是加载的“.jar”列表。
    • 上面的输出详细说明了打开的jar随后的类加载
    • 对,一开始我只阅读了最右边的“.jar”......我的错误。对不起!
    • 这是某种黑客行为吗?用户配置会调整这个详细级别吗?
    • 如果每个java进程有一个JVM,即一台机器上可以运行多个JVM(java进程),那么你的命令输出对应的是哪个JVM?
    【解决方案2】:

    使用Reflections 库,很简单:

    Reflections reflections = new Reflections("my.pkg", new SubTypesScanner(false));
    

    这将扫描包含 my.pkg 包的 url/s 中的所有类。

    • false 参数表示-不排除默认排除的Object 类。
    • 在某些情况下(不同的容器),您可能会传递 classLoader 以及参数。

    因此,获取所有类实际上是获取 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());
    

    【讨论】:

    • 您使用的是什么反射库版本?此示例不适用于反射 0.9.5。
    • 没关系,只是使用反射-0.9.9-RC1-uberjar 让它工作
    • 您只需要经历将另一个库添加到您的项目中的痛苦。当 Java 中的语言有问题时,你不能使用“简单”这个词。你给了我希望,认为这是 JDK 的标准反射。
    【解决方案3】:

    这个问题有多个答案,部分原因是问题含糊不清 - 标题是关于 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();
            }
        }
    
    }
    

    其中一个巧妙之处在于您可以选择要检查的任意类加载器。然而,如果类加载器类的内部发生变化,它可能会中断,因此它被用作一次性诊断工具。

    【讨论】:

    • @eis 使用Thread.currentThread().getContextClassLoader()CPTest.class.getClassLoader() 有什么区别?您是否有理由更喜欢从 Thread 对象中获取 ClassLoader
    • @MichaelPlautz 区别在于另一个是线程上下文类加载器,另一个是给定类使用的类加载器。 java使用了很多不同的类加载器,我不会在这个评论空间中详细说明。原因是假设线程上下文类加载器父关系将包括整个JVM,而类加载器可能不包括。
    • 你可以通过ClassLoader.class.getDeclaredField("classes")来避免while循环。
    • @NathanH 在android中没有这样的字段。比较 android classloaderjava se classloader
    • @IrfanLatif 我对 android 没有太多经验,所以无法帮助你。一般来说,一个新问题是针对新主题的:)
    【解决方案4】:

    上述方法的另一种方法是使用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 加载了给定类的信息。

    【讨论】:

      【解决方案5】:

      我还建议您编写一个-javagent 代理,但使用getAllLoadedClasses 方法而不是转换任何类。

      要与您的客户端代码(普通 Java 代码)同步,请创建一个套接字并通过它与代理通信。然后,您可以在需要时触发“列出所有类”方法。

      【讨论】:

        【解决方案6】:

        如果您已经知道包顶级路径,一种方法是使用OpenPojo

        final List<PojoClass> pojoClasses = PojoClassFactory.getPojoClassesRecursively("my.package.path", null);
        

        然后您可以浏览列表并执行您想要的任何功能。

        【讨论】:

          【解决方案7】:

          该程序将打印所有类及其物理路径。如果您需要分析从任何 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();
                  }
              }
          }
          

          【讨论】:

            【解决方案8】:

            JVM中加载的所有类的列表

            Oracle doc,您可以使用-Xlog 选项,该选项可以写入文件。

            java -Xlog:class+load=info:classloaded.txt
            

            【讨论】:

              【解决方案9】:

              您可能能够获得通过类加载器加载的类的列表,但这不包括您尚未加载但在类路径中的类。

              要获得类路径中的所有类,您必须执行类似于第二个解决方案的操作。如果您确实想要当前“已加载”的类(换句话说,您已经引用、访问或实例化的类),那么您应该细化您的问题以表明这一点。

              【讨论】:

                【解决方案10】:

                在 JRockit JVM 下运行您的代码,然后使用 JRCMD &lt;PID&gt; print_class_summary

                这将输出所有加载的类,每行一个。

                【讨论】:

                  【解决方案11】:

                  好吧,我所做的只是列出类路径中的所有文件。这可能不是一个光荣的解决方案,但它可靠地工作,并为我提供了我想要的一切,等等。

                  【讨论】:

                  • 能给个代码sn-p吗?这样其他人就可以从您的解决方案中受益。谢谢
                  • @jtzero:我不知道 OP 使用了什么,但对我来说 System.getProperty("java.class.path") 效果很好。
                  • 在尝试回答更多问题之前,请阅读How do I write a good answer?
                  猜你喜欢
                  • 2013-05-09
                  • 1970-01-01
                  • 2014-04-28
                  • 1970-01-01
                  • 2015-11-02
                  • 1970-01-01
                  • 1970-01-01
                  • 2020-02-19
                  • 1970-01-01
                  相关资源
                  最近更新 更多