NoClassDefFoundError和ClassNotFoundException之间的区别是什么?

是什么导致它们被抛出?如何解决这些问题?

在修改现有代码以包含新的jar文件时,我经常遇到这些可抛出的文件。 我已经在客户端和服务器端为一个通过webstart分发的java应用程序击中了它们。

我发现的可能原因是:

未包含在代码客户端build.xml中的包 我们正在使用的新jar缺少运行时类路径 版本与上一个jar冲突

当我今天遇到这些问题时,我采取了一种反复尝试的方法来让事情正常运行。我需要更多的清晰和理解。


当前回答

当ClassLoader没有找到所报告的类时,将抛出ClassNotFoundException。这通常意味着类从CLASSPATH中缺失。这也可能意味着这个类正试图从父类加载器中加载的另一个类中加载,因此子类加载器中的类是不可见的。在应用程序服务器等更复杂的环境中工作时,有时会出现这种情况(WebSphere因此类类加载器问题而臭名昭著)。

人们经常会混淆java.lang.NoClassDefFoundError和java.lang.ClassNotFoundException,但是它们之间有一个重要的区别。例如,一个异常(实际上是一个错误,因为java.lang.NoClassDefFoundError是java.lang.Error的子类)像

java.lang.NoClassDefFoundError:
org/apache/activemq/ActiveMQConnectionFactory

does not mean that the ActiveMQConnectionFactory class is not in the CLASSPATH. Infact its quite the opposite. It means that the class ActiveMQConnectionFactory was found by the ClassLoader however when trying to load the class, it ran into an error reading the class definition. This typically happens when the class in question has static blocks or members which use a Class that's not found by the ClassLoader. So to find the culprit, view the source of the class in question (ActiveMQConnectionFactory in this case) and look for code using static blocks or static members. If you don't have access the the source, then simply decompile it using JAD.

在检查代码时,假设你发现如下一行代码,确保类SomeClass在CLASSPATH中。

private static SomeClass foo = new SomeClass();

提示:要找出一个类属于哪个jar,您可以使用网站jarFinder。这允许您使用通配符指定类名,并在其jar数据库中搜索该类。Jarhoo允许你做同样的事情,但它不再免费使用。

如果您想在本地路径中定位类属于哪个jar,可以使用jarscan (http://www.inetfeedback.com/jarscan/)之类的实用程序。您只需指定您想要定位的类以及您希望它开始在jar和zip文件中搜索类的根目录路径。

其他回答

NoClassDefFoundError基本上是一个链接错误。当您尝试实例化一个对象(静态地使用“new”),并且在编译期间没有找到它时,就会发生这种情况。

ClassNotFoundException更为通用,是当您尝试使用不存在的类时的运行时异常。例如,你在函数中有一个参数接受一个接口,有人传入一个实现该接口的类,但你没有访问该类的权限。它还涵盖了动态类加载的情况,例如使用loadClass()或class . forname()。

通过名称本身,我们可以很容易地识别一个来自异常,另一个来自错误。

Exception:程序执行过程中出现异常。程序员可以通过try catch块处理这些异常。我们有两种类型的异常。在编译时抛出的已检查异常。在运行时抛出的运行时异常,这些异常通常是由于糟糕的编程而发生的。

错误:这些根本不是异常,它超出了程序员的范围。这些错误通常由JVM抛出。


图片来源

的区别:

ClassNotFoundException:

类装入器无法验证链接中的字节代码。 ClassNotFoundException是一个受控异常,当应用程序试图通过类的全限定名加载类,但在类路径上找不到类的定义时,就会发生这种异常。 当使用ClassLoader.loadClass()、class . forname()和ClassLoader.findSystemClass()在运行时提供类名,涉及显式加载类时,就会出现ClassNotFoundException。

NoClassDefFoundError:

类装入器无法解析链接中的类引用。 NoClassDefFoundError是一个从LinkageError类派生的错误,这是一个致命错误。当JVM试图使用new关键字实例化一个类或使用方法调用加载一个类时,无法找到类的定义,就会发生这种情况。 NoClassDefFoundError是由于类的方法调用或任何变量访问而隐式加载类的结果。


相似之处:

NoClassDefFoundError和ClassNotFoundException都与类在运行时不可用有关。 ClassNotFoundException和NoClassDefFoundError都与Java类路径相关。

ClassNotFoundException和NoClassDefFoundError的区别

ClassNotFoundException是一个检查异常,当我们告诉JVM使用class . forname()或ClassLoader.findSystemClass()或ClassLoader.loadClass()方法根据类的字符串名加载类,并且在类路径中没有找到所提到的类时,就会发生这种异常。

大多数情况下,当您试图运行应用程序而不使用所需的JAR文件更新类路径时,会发生此异常。例如,你可能已经看到这个异常时,做JDBC代码连接到你的数据库,即mysql,但你的类路径没有JAR。

NoClassDefFoundError发生错误当JVM试图加载一个特定的类的代码执行(作为一个正常的一部分方法调用或使用新的关键字创建一个实例的一部分),这类在类路径中不存在,但在编译时,因为为了执行程序需要编译它,如果你正在使用一个类不存在编译器会提高编译错误。

下面是简短的描述

你可以阅读关于ClassNotFoundException Vs NoClassDefFoundError的一切来了解更多细节。

与Java API规范的区别如下。

ClassNotFoundException:

当应用程序尝试时抛出 通过类的字符串加载类 名称使用: 类class中的forName方法。 类ClassLoader中的findSystemClass方法。 类ClassLoader中的loadClass方法。 但是没有类的定义 可以找到指定的名称。

NoClassDefFoundError:

抛出如果Java虚拟机或 ClassLoader实例尝试加载 在类的定义中(作为部分) 的正常方法调用或作为的一部分 方法创建一个新实例 表达式)和没有定义的 可以找到类。 搜索的类定义 当前执行时已存在 类被编译,但是定义 再也找不到。

因此,当源代码成功编译时,NoClassDefFoundError似乎发生了,但在运行时,没有找到所需的类文件。这可能发生在JAR文件的分发或生产过程中,其中没有包括所需的所有类文件。

至于ClassNotFoundException,它似乎起源于试图在运行时对类进行反射调用,但程序试图调用的类并不存在。

两者之间的区别是一个是错误,另一个是异常。NoClassDefFoundError是一个错误,它源于Java虚拟机在查找期望查找的类时遇到问题。期望在编译时工作的程序不能运行,因为没有找到类文件,或者与在编译时生成或遇到的类文件不相同。这是一个非常严重的错误,因为该程序不能由JVM启动。

另一方面,ClassNotFoundException是一个异常,所以它在某种程度上是预期的,并且是可以恢复的。使用反射是很容易出错的(因为有一些预期,事情可能不会像预期的那样进行。没有编译时检查来查看所有必需的类是否存在,因此任何查找所需类的问题都会在运行时出现。