2

我是一个经验丰富的程序员,我遇到了这个非常奇怪的问题,到目前为止我还无法解决。我收到此错误:

Exception in thread "Thread-0" java.lang.NoClassDefFoundError: rec/MiscIO
        at rec.RECTool.run(RECTool.java:264)
        at java.lang.Thread.run(Unknown Source)
Caused by: java.lang.ClassNotFoundException: rec.MiscIO
        at java.net.URLClassLoaderrun(Unknown Source)
        at java.net.URLClassLoaderrun(Unknown Source)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        ... 2 more

这是相关的行:

File savedTo = MiscIO.copyJar(root);

我不明白的是,这是一个小应用程序,并且正在使用的类在同一个包中。除了类名和内容外,基本相同。但是,我只在我的一些课程中特别得到了错误,而不是全部。我在错误发生之前使用这些行,它们在同一个包和所有内容中。

Logger.log("Found new root: " + newRoots[i0]);
...
FileLocker locker = new FileLocker();
...
locker.lock(PathManager.getJar());

所以三个类的设置与出错的一个基本相同,只有一个抛出异常。这对我来说毫无意义。

我尝试了各种参数来运行 jar,包括修改类路径和直接调用主类而不是使用 jar 命令,但无济于事,例如:

java -jar <jarname>
java -cp . <jarname>
java -cp <jarname> -jar <jarname>
java -cp <jarname> <main class here>

但是,如果我解压缩 jar 并运行程序,则不会发生错误。我认为这可能是 MiscIO 无法加载或静态初始化程序无法加载的某种依赖关系,但我的代码中都没有。

另外,我正在使用 Netbeans 生成 jar,并且清单设置正确。这是整个代码:

    package rec;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.RandomAccessFile;
import java.net.URL;

/**
 * @author Colby
 */
public class MiscIO {

    public static void copyJar(RandomAccessFile out) throws IOException {
        URL jarSource = RECTool.class.getProtectionDomain().getCodeSource().getLocation();

        InputStream in = null;
        try {
            in = jarSource.openStream();
            copyToRAF(in, out);

        } finally {
            if (in != null) {
                in.close();
            }
        }
    }

    public static File copyJar(File saveJarTo) throws IOException {
        URL jarSource = RECTool.class.getProtectionDomain().getCodeSource().getLocation();
        saveJarTo = new File(saveJarTo, "classlist.jar");

        InputStream in = null;
        OutputStream out = null;
        try {
            in = jarSource.openStream();
            out = new FileOutputStream(saveJarTo);
            copyStream(in, out);

        } finally {
            if (in != null) {
                in.close();
            }
            if (out != null) {
                out.close();
            }
        }
        return saveJarTo;
    }

    public static void copyStream(InputStream in, OutputStream out) throws IOException {
        int len;
        byte[] data = new byte[1024 * 8];
        while ((len = in.read(data)) != -1) {
            out.write(data, 0, len);
        }
        out.flush();
    }

    public static void copyToRAF(InputStream in, RandomAccessFile raf) throws IOException {
        raf.seek(0L);

        int len;
        byte[] data = new byte[1024 * 8];
        while ((len = in.read(data)) != -1) {
            raf.write(data, 0, len);
        }
    }

    public static void copyFile(File from, File to) throws IOException {
        FileInputStream in = null;
        FileOutputStream out = null;
        try {
            in = new FileInputStream(from);
            out = new FileOutputStream(to);
            copyStream(in, out);

        } finally {
            if (in != null) {
                in.close();
            }
            if (out != null) {
                out.close();
            }
        }
    }
}

最后一点,我认为这可能是我机器上安装的 Java 版本冲突,或者某些侥幸的 x86/x64 冲突,所以我卸载了所有 Java 发行版,只安装了一个架构中的最新版本,问题仍然存在。你们有谁以前见过这样的事情吗?

4

3 回答 3

2

根据您所说的,最可能的解释是您错误地创建了 JAR 文件。

运行jar -tvf <jarname>。它应该将 MiscIO 类显示为“rec/MiscIO.class”。如果没有,那么类加载器将无法找到它。

(还有其他不太可能的解释;例如,涉及自定义类加载器和/或 JVM 试图加载之前类加载失败的类。)


作为记录,冲突的 Java 版本不会导致这些确切症状出现问题。(或者至少,>>I<< 看不出它怎么会……)


您的代码表明它是更复杂和晦涩的东西......并且可能依赖于平台。

于 2012-11-22T03:40:53.850 回答
2

另一个不太可能的解释可能是Thread Context ClassLoader

我看到Exception发生在一个新的Thread中,所以可能是Context Class Loader和调用Thread不一样。这取决于创建新线程的代码,以及它是否位于找不到类的相同类路径中,如以下 Stack Overflow 答案中所述:

每个类都将使用它自己的类加载器来加载其他类。因此,如果 ClassA.class 引用 ClassB.class 那么 ClassB 需要位于 ClassA 的类加载器的类路径上,或者它的父级。

线程上下文类加载器是当前线程的当前类加载器。可以从 ClassLoaderC 中的类创建对象,然后将其传递给 ClassLoaderD 拥有的线程。在这种情况下,如果对象想要加载它自己的类加载器上不可用的资源,则需要直接使用 Thread.currentThread().getContextClassLoader() 。

(摘自: 线程的上下文类加载器和普通类加载器的区别

你可以做一个排除这个的测试是在你的应用程序的主线程中断言它Class.forName("rec.MiscIO")不为空。

于 2012-11-22T03:50:55.583 回答
2

兄弟,你的文件柜是做什么用的?

1    FileLocker locker = new FileLocker();
2    locker.lock(PathManager.getJar());
3    File savedTo = MiscIO.copyJar(root);

如果 line[2] 独占锁定包含 的 jar MiscIO.class,则 line[3] 无法加载 MiscIO 类。

于 2012-11-22T04:01:08.813 回答