繁体   English   中英

从课程中获取完整的课程路径

[英]getting full classpath from a class

我正在寻找一种实用程序方法,以便给定一个类将返回在外部运行该类所需的完整类路径。 这意味着该类所在的jar以及它使用的所有类的jar(或文件夹)。

更新:有一些工具可以分析.class文件来查找依赖关系。 这不是我想要的。 我正在寻找在已加载的类上使用Java的反射API的东西。 我将解决一些分析字节码的问题,如果它可以递归地进入通过类加载器找到的类中

我认为这是不可能的。 当然,反射API不支持它。

您可以找到一个类classloader,但不能找到:

  • 类加载器的哪些可能的JAR文件和目录包含该类,
  • 该类的静态依赖项是什么,或者
  • 类的动态依赖关系是什么? 例如,使用Class.forName()加载的内容或其依赖项。

实际上,这有点夸大了事情:

  • 从理论上讲,如果可以找出类路径是什么,则可以弄清楚哪些类来自哪个JAR。 可能有多种方法可以从类加载器中挖掘出来。
  • 从理论上讲,您可以弄清楚类的依赖项是什么,但是您一直在使用BCEL(例如)在类的字节码文件中进行挖掘以找出答案。
  • 从理论上讲,如果您准备编写自己的类加载器,则可以找出动态加载的内容。 可能有可能使用堆栈框架的一些多方面分析将其链接回启动加载的类。

但这都是极其复杂的,我希望它在某些情况下不可靠。

反思不会对您有太大帮助。 您将需要分析字节码以找到依赖关系。

更新:

那好吧 我使用的是我几年前制作的库,您可以在此处下载。

如下代码:

package classdep;

import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import org.jedo.classfile.ClassFile;
import org.jedo.classfile.ConstantPool;

public class Main {

    public static void main(String[] args) {
        try {
            ClassLoader cl = Thread.currentThread().getContextClassLoader();
            List<String> classes = new ArrayList<String>();
            classes.add(args[0].replace('.', '/'));
            for (int i = 0; i < classes.size(); ++i) {
                String className = classes.get(i);
                URL url = cl.getResource(className + ".class");
                if (url == null) {
                    System.out.println("--- class not found " + className);
                } else {
                    System.out.println(url);
                    ClassFile classFile = new ClassFile();
                    InputStream in = url.openStream();
                    try {
                        classFile.load(in);
                    } finally {
                        in.close();
                    }
                    ConstantPool cp = classFile.getConstantPool();
                    for (String name: cp.getClassNames()) {
                        if (!classes.contains(name)) {
                            classes.add(name);
                        }
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

将为您提供类的所有依赖关系。 当应用于org.jedo.classfile.ClassFile时,它将产生以下输出:

file:/D:/projects/casagrande/jedo/build/classes/org/jedo/classfile/ClassFile.class
file:/D:/projects/casagrande/jedo/build/classes/org/jedo/classfile/ConstantPool.class
file:/D:/projects/casagrande/jedo/build/classes/org/jedo/classfile/FieldInfo.class
file:/D:/projects/casagrande/jedo/build/classes/org/jedo/classfile/MethodInfo.class
file:/D:/projects/casagrande/jedo/build/classes/org/jedo/classfile/AttributeInfo.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/File.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/FileInputStream.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/DataInputStream.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/StreamCorruptedException.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/FileOutputStream.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/DataOutputStream.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/lang/StringBuilder.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/lang/StringBuffer.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/lang/Object.class
jar:file:/C:/Program%20Files/Java/jdk1.6.0_18/jre/lib/rt.jar!/java/io/IOException.class
...

其次是很多系统类。 您需要过滤掉系统类,并解析其他URL以提取.jar文件(如果它是jar:url)或目录(如果它是文件:url)。

在某些情况下,您无法在使用课堂之前确定这一点。

这并不总是已知的。 例如,可以在运行时动态创建一个类,然后使用自定义ClassLoader

我不相信Java存储此信息。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM