NoClassDefFoundError与ClassNotFoundException的区别

本文深入探讨了Java运行时中遇到的两类常见错误:ClassNotFoundException和NoClassDefFoundError的区别与解决策略。通过分析类加载过程和错误发生的场景,为开发者提供了准确诊断和有效解决问题的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

[b]相同点:[/b]
1、NoClassDefFoundError和ClassNotFoundException都是类运行时出错。

2、都和classpath有关。

[b]不同点:[/b]
1、ClassNotFoundException继承java.lang.Exception,是一种受检异常(checked exception),需要显式地使用try/catch来进行处理。通常需要确保需要的类已经在classpath上了。

--------------------------------
而NoClassDefFoundError继承java.lang.LinkageError,是一种错误(Error)。

2、ClassNotFoundException发生在类加载器在classpath上找不到相关类的时候,通常会使用Class.forName()或ClassLoader.loadClass()或ClassLoader.findSystemClass()。

有时我们会主观地认为类已经在classpath上了,但其实不是,例如jar包中的manifest文件的优先级比当前classpath或-cp选项指定的classpath的优先级都要高,就有可能发生ClassNotFoundException。

还有一种也会发生ClassNotFoundException的场景就是使用两个类加载器,一个类加载器尝试着访问已经被另一个类加载器加载的类也会导致ClassNotFoundException。

--------------------------------
而NoClassDefFoundError相对来说更难诊断出原因。

有时会发生在运行期依赖的类变更了或被删除了,例如下面这个例子,Test类中依赖Test1类,在编译时Test1存在,
[img]http://dl2.iteye.com/upload/attachment/0101/3771/f8980746-9046-3efd-b307-0d5e38590a50.png[/img]
编译成功后,删除Test1.class文件,再运行Test类时,就会报错:
Exception in thread "main" java.lang.NoClassDefFoundError: Test
        at Test1.main(Test1.java:5)
Caused by: java.lang.ClassNotFoundException: Test
        at java.net.URLClassLoader$1.run(Unknown Source)
        at java.net.URLClassLoader$1.run(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)
        ... 1 more

NoClassDefFoundError还可能发生在类的静态代码块中抛出了异常,例如:
[img]http://dl2.iteye.com/upload/attachment/0101/3774/07e4c260-7f27-31e1-a7e8-bf8b6f84b32c.jpg[/img]
配置文件中没有PROM_REDIS_MAXACTIVE项,是null,导致类的静态代码块出错,从而引起NoClassDefFoundError:
java.lang.NoClassDefFoundError: com.jag.StringSingleCacheClient (initialization failure)
at java.lang.J9VMInternals.initialize(J9VMInternals.java:140)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值