极客时间杨晓峰-Java核心技术36讲-第2讲:Exception和Error有什么区别?运行时异常与一般异常有什么区别?

极客时间杨晓峰-Java核心技术36讲-笔记记录

第2讲:Exception和Error有什么区别?运行时异常与一般异常有什么区别?

2.1格式化回答

先给出该题标准的格式化回答:

Exception和Error都是继承了Throwable类,在Java中只有Throwable类型的实例才可以被抛出(throw)或者捕获(catch),它是异常处理机制的基本组成类型。

Exception和Error体现了Java平台设计者对不同异常情况的分类。Exception是程序正常运行中,可以预料的意外情况,可能并且应该被捕获,进行相应处理。

Error是指在正常情况下,不大可能出现的情况,绝大部分的Error都会导致程序(比如JVM自身)处于非正常的、不可恢复状态。既然是非正常情况,所以不便于也不需要捕获,常见的比如OutOfMemoryError之类,都是Error的子类。

Exception又分为可检查(checked)异常和不检查(unchecked)异常,可检查异常在源代码里必须显式地进行捕获处理,这是编译期检查的一部分。前面我介绍的不可查的Error,是Throwable不是Exception。不检查异常就是所谓的运行时异常,类似 NullPointerException、ArrayIndexOutOfBoundsException之类,通常是可以编码避免的逻辑错误,具体根据需要来判断是否需要捕获,并不会在编译期强制要求。

2.2该题考点分析与相关扩展方向

分析Exception和Error的区别,是从概念角度考察了Java异常处理机制,我们需要理解Throwable、Exception、Error的设计和分类,掌握那些应用最为广泛的子类,学习如何自定义异常。给出一个简单的类图方便概念记忆。

img

首先,常见的异常子类是面试常问的知识点,比如NoClassDefFoundError和ClassNotFoundException有什么区别?

NoClassDefFoundError是一个错误(Error),而ClassNOtFoundException是一个异常,在Java中对于错误和异常的处理是不同的,我们可以从异常中恢复程序但却不应该尝试从错误中恢复程序。

ClassNotFoundException的产生原因:Java支持使用Class.forName方法来动态地加载类,任意一个类的类名如果被作为参数传递给这个方法都将导致该类被加载到JVM内存中,如果这个类在类路径中没有被找到,那么此时就会在运行时抛出ClassNotFoundException异常。解决该问题需要确保所需的类连同它依赖的包存在于类路径中,常见问题在于类名书写错误。另外还有一个导致ClassNotFoundException的原因就是:当一个类已经某个类加载器加载到内存中了,此时另一个类加载器又尝试着动态地从同一个包中加载这个类。通过控制动态类加载过程,可以避免上述情况发生。

NoClassDefFoundError的产生原因:如果JVM或者ClassLoader实例尝试加载(可以通过正常的方法调用,也可能是使用new来创建新的对象)类的时候却找不到类的定义。要查找的类在编译的时候是存在的,运行的时候却找不到了。这个时候就会导致NoClassDefFoundError.造成该问题的原因可能是打包过程漏掉了部分类,或者jar包出现损坏或者篡改。解决这个问题的办法是查找那些在开发期间存在于类路径下但在运行期间却不在类路径下的类。

其次,理解Java语言中操作Throwable的元素和实践技巧。

​ A.不要推诿或延迟处理异常,就地解决最好,并且需要实实在在的进行处理,而不是只捕捉,不动作。

​ B.一个函数尽管抛出了多个异常,但是只有一个异常可被传播到调用端。最后被抛出的异常时唯一被调用端接收的异常,其他异常都会被吞没掩盖。如果调用端要知道造成失败的最初原因,程序之中就绝不能掩盖任何异常。

​ C.不要在fnally代码块中处理返回值。

​ D.按照我们程序员的惯性认知:当遇到return语句的时候,执行函数会立刻返回。但是,在Java语言中,如果存在fnally就会有例外。除了return语句,try代码块中的break或continue语句也可能使控制权进入fnally代码块。

​ E.请勿在try代码块中调用return、break或continue语句。万一无法避免,一定要确保fnally的存在不会改变函数的返回值。

​ F.函数返回值有两种类型:值类型与对象引用。对于对象引用,要特别小心,如果在fnally代码块中对函数返回的对象成员属性进行了修改,即使不在fnally块中显式调用return语句,这个修改也会作用于返回值上。

​ G.勿将异常用于控制流。

​ I.如无必要,勿用异常。

-------------本文结束感谢您的阅读-------------
zouzou wechat
欢迎您扫一扫上面的微信号,添加我的私人微信!
坚持原创技术分享,您的支持将鼓励我继续创作!