有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java如何强制类。forName只看罐子里面?

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

  1. 我的应用程序。罐子
  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外

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

共 (1) 个答案

  1. # 1 楼答案

    在这个场景中,类装入器找到了B,但它不在jar中,您还可以检查

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

    或者

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

    如果其中任何一个是真的,那么B是从MyApp加载的。罐子

    比如说:

    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");
    }