ClassNotFoundException 与 NoClassDefFoundError 到底有什么区别?

ClassNotFoundException 与 NoClassDefFoundError 到底有什么区别?

Photo By Instagram sooyaaa

问题 12

你在开发中碰到过 ClassNotFoundException 和 NoClassDefFoundError 吗? 它们有什么区别?

我的答案

首先这俩个错误都代表着 JVM 无法找到相关的类而抛出的错误,但是它们发生的场景,以及类型却截然不同。

类型

从类型上来说 ClassNotFoundException 是 Exception 系的,而 NoClassDefFoundError 是 Error 系的,虽然他们都是 Throwable 的子孙,但是使用场景却截然不同。

Error 系的错误一旦抛出,则代表着这个错误是无法修复的,应用程序不应当去捕捉这个错误,因为这个类型的错误正常情况下永远也不会出现。

Exception 系的轻微一点,这种错误允许我们在应用程序中处理,是一种可以预知的错误。

发生场景

ClassNotFoundException 一般会在我们的应用程序去试图加载一个类的时候,发现当前类路径中无法找到该类的 class 文件而抛出的,例如如下几个方法:

Class 的 forName 方法

ClassLoader 的 findSystemClass 和 loadClass 方法

虽然 NoClassDefFoundError 也是在 JVM 试图加载一个类的时候发现类不存在然后抛出的错误,但是它是在我们使用 new 来实例化一个对象的时候抛出的。

如下我们做了一个小测试,定义了一个 Solution 类,然后在 Main 类中使用它,当 2 个类都编译完成以后,我们删除掉 Solution 类的 class 文件,然后运行 Main 类:

1 public class   Main   {

2

3 public   static   void   main (String[] args)   throws  ClassNotFoundException  {

4 try {

5 Solution solution =  new Solution();

6catch (Throwable e) {

7 e.printStackTrace();

8 }

9

10 try {

11 Class.forName( "jvm.Solution" );

12catch (Exception e) {

13 e.printStackTrace();

14 }

15 System.out.println( "over" );

16 }

17 }

18

19 public class   Solution   {

20

}

此时我们会得到如下异常信息:

 1java.lang.NoClassDefFoundError: jvm/Solution
 2       at jvm.Main.main(Main.java:12)
 3Caused by: java.lang.ClassNotFoundException: jvm.Solution
 4       at java.net.URLClassLoader$1.run(URLClassLoader.java:372)
 5       at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
 6       at java.security.AccessController.doPrivileged(Native Method)
 7       at java.net.URLClassLoader.findClass(URLClassLoader.java:360)
 8       at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
 9       at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
10       at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
11      ... 1 more
12java.lang.ClassNotFoundException: jvm.Solution
13       at java.net.URLClassLoader$1.run(URLClassLoader.java:372)
14       at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
15       at java.security.AccessController.doPrivileged(Native Method)
16       at java.net.URLClassLoader.findClass(URLClassLoader.java:360)
17       at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
18       at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)

你会发现,使用 new 来初始化对象时候,抛出来的最外层是 NoClassDefFoundError 错误,然而它的底层还是因为 ClassNotFoundException 异常造成的。 而使用 Class.forName 去加载类呢,则会直接抛出 ClassNotFoundException 异常。

如上即为 ClassNotFoundException 和 NoClassDefFoundError 的区别,你 get 到了吗?

以上即为昨天的问题的答案,小伙伴们对这个答案是否满意呢?欢迎留言和我讨论。

又要到年末了,你是不是又悄咪咪的开始看机会啦。 为了广大小伙伴能充足电量,能顺利通过 BAT 的面试官无情三连炮,我特意推出大型刷题节目。 每天一道题目,第二天给答案,前一天给小伙伴们独立思考的机会。

ClassNotFoundException 与 NoClassDefFoundError 到底有什么区别?

点下“在看”,鼓励一下?

原文 

https://mp.weixin.qq.com/s/C3LdKfnjeajYR6e-x_Idrw

本站部分文章源于互联网,本着传播知识、有益学习和研究的目的进行的转载,为网友免费提供。如有著作权人或出版方提出异议,本站将立即删除。如果您对文章转载有任何疑问请告之我们,以便我们及时纠正。

PS:推荐一个微信公众号: askHarries 或者qq群:474807195,里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

转载请注明原文出处:Harries Blog™ » ClassNotFoundException 与 NoClassDefFoundError 到底有什么区别?

赞 (0)
分享到:更多 ()

评论 0

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址