java - 如何强制 Class.forName 仅查看 jar 内部?

标签 java ant jar classpath classloader

请想象一个包含两个文件的文件夹

  1. MyApp.jar
  2. B.class(真实路径很长)

jar 里面有一些代码来检查 B.class 是否存在内部 jar 文件。

try {
    Class.forName("B");
    System.out.println("exists");
} catch (Exception ignored) {
    System.out.println("does not exist");
}

但是即使 B.class 不在 jar 内,上面的代码也不会抛出异常,因为 B.class 存在于 jar 之外。

该 jar 是使用 Eclipse 中的 Ant 生成的。所以我认为类路径可能是原因

 <manifest>
     <attribute name="Main-Class" value="org.eclipse.jdt.internal.jarinjarloader.JarRsrcLoader"/>
     <attribute name="Rsrc-Main-Class" value="org.client.Client"/>
     <attribute name="Class-Path" value="."/>
     <attribute name="Rsrc-Class-Path" value="./ many_jar_here.jar"/>
  </manifest>

所以我只改变了这样的类路径

<attribute name="Class-Path" value=""/>

但是现在它给出了这样的错误:

 Exception in thread "main" java.lang.NoClassDefFoundError: B
        at java.lang.Class.forName0(Native Method)
        at java.lang.Class.forName(Class.java:264)
        at org.eclipse.jdt.internal.jarinjarloader.JarRsrcLoader.main(JarRs
der.java:56)
Caused by: java.lang.ClassNotFoundException: B
        at java.net.URLClassLoader$1.run(URLClassLoader.java:366)
        at java.net.URLClassLoader$1.run(URLClassLoader.java:355)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(URLClassLoader.java:354)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:423)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:356)
        ... 3 more

最佳答案

在这种情况下,类加载器找到 B 但它不在 jar 中,您还可以检查是否

B.class.getProtectionDomain().getCodeSource() == ClassKnownToBeInJar.class.getProtectionDomain().getCodeSource()

B.class.getProtectionDomain().getCodeSource().getLocation().getPath().contains("MyApp.jar")

如果其中任何一个为真,则 B 是从 MyApp.jar 加载的。

所以,像这样:

try {
    Class bClass = Class.forName("B");
    System.out.println("exists");
    if (bClass.getProtectionDomain().getCodeSource().getLocation().getPath().contains("MyApp.jar")) {
        System.out.println("class loaded from MyApp.jar");
    } else {
        System.out.println("class not loaded from MyApp.jar");
    }
} catch (Exception ignored) {
    System.out.println("does not exist");
}

关于java - 如何强制 Class.forName 仅查看 jar 内部?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13909100/

相关文章:

java - 如何使用 eclipse 将类路径添加到导出的可运行 java .jar 文件

java - Maven多项目库依赖

java - 编译器的模糊参数,当它对我来说不模糊时

java - 如何使用 GUI 和 CLI 构建 Java 应用程序?

java - 如何将 ojdbc jar 添加到我的项目中并使用它

java - 只在 JAR 中保留某些类的 Ant 任务 - 具有依赖性检查?

java - 检查路径在java中是否有效或合法

java - 绘画应用程序中撤消/重做的命令模式

groovy - 带有逗号分隔文件的 AntBuilder (groovy) (FileSet "include")

java - 为什么 Ant 编译的 Java 源代码的目标 Java 版本具有未定义的方法