java - javaagent jar 在 bootclasspath 中的位置

标签 java classloader bytecode instrumentation javaagents

我有一个 javaagent jar,我使用

Boot-Class-Path: myagent.jar

在 MANIFEST.MF 文件中。

我需要找出 jar 所在的文件系统上的目录。

但是针对此描述的方法here似乎对我不起作用:

 new File(MyClass.class.getProtectionDomain().getCodeSource().getLocation().toURI().g­etPath());

在这种情况下,ProtectionDomain.getCodeSource() 返回 null。我猜这是因为 jar 已放在引导类路径中。因此,我也无法执行 MyClass.getClassLoader() 来获取资源位置。

我正在使用 Java 6。

谁能告诉我如何获取 jar 的位置?

最佳答案

您可以使用系统类加载器在引导类路径中查找类。比如这个

System.out.println(
  ClassLoader.getSystemClassLoader().getResource("java/lang/String.class")
);

会打印出类似的东西,

jar:file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/rt.jar!/java/lang/String.class

要查找 MyClass.class 在磁盘上的位置,请执行

String urlString = ClassLoader
 .getSystemClassLoader()
 .getResource("com/my/package/MyClass.class")
 .toString();

urlString = urlString.substring(urlString.indexOf("file:"), urlString.indexOf('!'));
URL url = new URL(urlString);
File file = new File(url.toURI());
System.out.println(file);
System.out.println(file.exists());

2020-06-29 更新,作者:kriegaex:我不想为这个老问题写一个新答案,而是想增强或更新这个非常好的答案,同时考虑带有空格和 Java 9+ 模块的帐户路径。

这是一个将类文件路径作为 File 实例返回的方法。如果类文件是 JAR 或 Java 运行时模块的一部分(URL 以协议(protocol) jrt: 开头),它只会返回 JAR 或 JMOD 文件的路径,这是您可能不想要的, 但它只是一个展示,您可以根据自己的喜好进行编辑。当然这仍然是 hacky,例如不考虑从 Web URL 而不是从文件加载的类,但你明白了。

public static File getFileForClass(String className) {
  className = className.replace('.', '/') + ".class";
  URL classURL = ClassLoader.getSystemClassLoader().getResource(className);
  if (classURL == null)
    return null;
  System.out.println("Class file URL: " + classURL);

  // Adapt this if you also have '.war' or other archive types
  if (classURL.toString().contains(".jar!")) {
    String jarFileName = classURL.getPath().replaceFirst("!.*", "");
    System.out.println("Containing JAR file: " + jarFileName);
    try {
      return new File(new URL(jarFileName).toURI());
    }
    catch (URISyntaxException | MalformedURLException e) {
      throw new RuntimeException(e);
    }
  }

  if (classURL.getProtocol().equals("jrt")) {
    String jrtModule = classURL.getFile().replaceFirst("/([^/]+).*", "$1");
    System.out.println("Target class is part of Java runtime module " + jrtModule);
    String jmodName = System.getProperty("java.home") + "/jmods/" + jrtModule + ".jmod";
    System.out.println("Containing Java module file: " + jmodName);
    return new File(jmodName);
  }
  try {
    return new File(classURL.toURI());
  }
  catch (URISyntaxException e) {
    throw new RuntimeException(e);
  }
}

当我从我的一个 IntelliJ IDEA 项目中调用它时,JDK 14 安装在一个包含空格的路径下,并故意添加一个 JAR 也包含一个包含用于测试的空格的路径,这段代码......

public static void main(String[] args) throws IOException {
  Instrumentation instrumentation = ByteBuddyAgent.install();
  instrumentation.appendToSystemClassLoaderSearch(
    new JarFile("C:/Program Files/JetBrains/IntelliJ IDEA 2018.3/lib/idea_rt.jar")
  );
  Stream
    .of(
      Weaver.class.getName(),
      String.class.getName(),
      ByteBuddy.class.getName(),
      "com.intellij.execution.TestDiscoveryListener"
    )
    .forEach(className -> System.out.printf("Found file: %s%n%n", getFileForClass(className)));
}

...产生这个控制台输出:

Class file URL: file:/C:/Users/alexa/Documents/java-src/Sarek/sarek-aspect/target/classes/dev/sarek/agent/aspect/Weaver.class
Found file: C:\Users\alexa\Documents\java-src\Sarek\sarek-aspect\target\classes\dev\sarek\agent\aspect\Weaver.class

Class file URL: jrt:/java.base/java/lang/String.class
Target class is part of Java runtime module java.base
Containing Java module file: C:\Program Files\Java\jdk-14.0.1/jmods/java.base.jmod
Found file: C:\Program Files\Java\jdk-14.0.1\jmods\java.base.jmod

Class file URL: jar:file:/C:/Users/alexa/.m2/repository/net/bytebuddy/byte-buddy/1.10.13/byte-buddy-1.10.13.jar!/net/bytebuddy/ByteBuddy.class
Containing JAR file: file:/C:/Users/alexa/.m2/repository/net/bytebuddy/byte-buddy/1.10.13/byte-buddy-1.10.13.jar
Found file: C:\Users\alexa\.m2\repository\net\bytebuddy\byte-buddy\1.10.13\byte-buddy-1.10.13.jar

Class file URL: jar:file:/C:/Program%20Files/JetBrains/IntelliJ%20IDEA%202018.3/lib/idea_rt.jar!/com/intellij/execution/TestDiscoveryListener.class
Containing JAR file: file:/C:/Program%20Files/JetBrains/IntelliJ%20IDEA%202018.3/lib/idea_rt.jar
Found file: C:\Program Files\JetBrains\IntelliJ IDEA 2018.3\lib\idea_rt.jar

关于java - javaagent jar 在 bootclasspath 中的位置,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5726930/

相关文章:

python - 重新编译 Python 字节码指令

java - 通过 ASM 跟踪方法依赖关系

java - 包含阿拉伯和西方字符的字符串连接

java - hibernate 中的@OneToMany

java - URLClassLoader.getResources ("") (空资源名称) 没有给出 jars 的根

java - 在运行时更新 JAR

api - api是否像bytcode一样提供多平台功能

java - 如何使用java查找操作系统中的用户名

java - 如何为数据库连接池找到一个合理的大小以及如何验证它?

java - eclipse插件中的类加载