如何强制 Class.forName 只查看 jar 内部



>请想象一个有两个文件的文件夹

  1. 我的应用.jar
  2. B.class(真实路径很长)
在jar内部

有一些代码来检查jar文件中是否存在B.class。

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

但即使 B.class 不在 jar 内部,上面的代码也不会引发异常,因为 B.class 存在于 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");
}

最新更新