ClassNotFoundException vs NoClassDefFoundError

2,328 阅读2分钟

相信java的开发者经常会遇到这两个异常,乍一看两个就长得很像,有时候真的是老虎,老鼠傻傻让人分不清楚。

1. 两者对比

类别 ClassNotFoundException NoClassDefFoundError
含义 找不到类定义 找不到类定义
继承 extends ReflectiveOperationException extends LinkageError
类型 反射操作引起的一种异常 一种链接错误
捕获 catch Exception/Throwable catch Error/Throwable
原因 通过反射根据指定类名无法找到类时,会引起该异常 编译时有类定义,运行时丢失类定义会引起该异常。运行时类定义丢失往往有两种情况:1)类不在ClassPath内,导致无法加载类定义,即由ClassNotFoundException引起。2)类初始化失败,导致加载不到类定义,此时会伴随着一个初始化异常(ExceptionInInitializerError)。

2. 两者关系

NoClassDefFoundError可由ClassNotFoundException引起。

3. 两者定义

NoClassDefFoundError doc

当JVM或类加载器去加载一个类定义(无论是通过普通的方法调用,还是用new语句去创建该类的实例所触发的类加载),如果该类的定义丢失,那么就会抛出NoClassDefFoundError. 被加载的类定义当前执行类编译时是存在的,但是再也找不到了。

ClassNotFound doc

当应用使用类的字符串名,通过以下方式尝试去加载类,但是根据指定的名称无法找到类定义时会抛出ClassNotFoundException: 1)Class.forName() 2)ClassLoader.findSystemClass() 3)ClassLoader.loadClass()

4. 实例

4.1 ClassNotFoundExcption
package qiweiTest;

public class ClassNotFoundExcptionTest {
	public static void main(String[] args) {
		Class.forName("anyNoexistClassName"); // 任何一个不存在的className
	}
}
4.2 NoClassDefError
4.2.1 类不在ClassPath引发的异常
package qiweiTest;

public class NoClassDefErrorTest {
     // 运行时,删除ClassPath中的ClassNotFoundExceptionTest.class
	public static void main(String[] args) {
		ClassNotFoundExceptionTest.class.getClass();
	}
}

输出:

Exception in thread "main" java.lang.NoClassDefFoundError: qiweiTest/ClassNotFoundExceptionTest
	at qiweiTest.NoClassDefErrorTest.main(NoClassDefErrorTest.java:9)
Caused by: java.lang.ClassNotFoundException: qiweiTest.ClassNotFoundExceptionTest
	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)
	... 1 more
4.2.2 类初始化失败引发的异常
package qiweiTest;

public class NoClassDefErrorTest2 {
       	public static void main(String[] args) {
		int tmp;
		try {
			tmp = InterClass.a;
		} catch (Throwable e) {
            // 异常类型:ExceptionInInitializerError
			e.printStackTrace();
		}
		try {
			tmp = InterClass.a;
		} catch (Throwable e) {
            // 异常类型:NoClassDefFoundError
			e.printStackTrace();
		}
	}

	static class InterClass {
		static int a;
		static {
			a = 1 / 0;
		}
	}

输出:

java.lang.ExceptionInInitializerError
	at qiweiTest.NoClassDefErrorTest2.main(NoClassDefErrorTest2.java:8)
Caused by: java.lang.ArithmeticException: / by zero
	at qiweiTest.NoClassDefErrorTest2$InterClass.<clinit>(NoClassDefErrorTest2.java:22)
	... 1 more
java.lang.NoClassDefFoundError: Could not initialize class qiweiTest.NoClassDefErrorTest2$InterClass
	at qiweiTest.NoClassDefErrorTest2.main(NoClassDefErrorTest2.java:13)