Exception 和 Error 的区别:
- Exception 和 Error 都是继承了 Throwable 类,在 Java 中只有 Throwable 类型的实例才可以被抛出(throw)或者捕获(catch),它是异常处理机制的基本组成类型。
- Exception 是程序正常运行中,可以预料的意外情况,可能并且应该被捕获,进行相应处理。
Exception 又分为可检查(checked)异常和不可检查异常。
可检查异常在源代码里必须显式地进行捕获处理,这是编译期检查的一部分。
不检查异常就是所谓的运行时异常,类似 NullPointerException、ArrayIndexOutOfBoundsException之类,通常是可以编码避免的逻辑错误,具体根据需要来判断是否需要捕获,并不会在编译期强制要求。
- Error 是指在正常情况下,不大可能出现的情况,绝大部分的 Error 都会导致程序(比如 JVM 自身)处于非正常的、不可恢复状态。
既然是非正常情况,所以不便于也不需要捕获,常见的比如 OutOfMemoryError 之类,都是 Error 的子类。
NoClassDefFoundError 和 ClassNotFoundException 的区别:
- NoClassDefFoundError是一个错误(Error)。
产生原因:
如果JVM或者ClassLoader实例尝试加载(可以通过正常的方法调用,也可能是使用new来创建新的对象)类的时候却找不到类的定义。要查找的类在编译的时候是存在的,运行的时候却找不到了,这个时候就会导致 NoClassDefFoundError。造成该问题的原因可能是打包过程漏掉了部分类,或者jar包出现损坏或者篡改。
解决方法:
查找那些在开发期间存在于类路径下但在运行期间却不在类路径下的类。
- ClassNotFoundException 是一个异常。
产生原因:
Java 支持使用 Class.forName 方法来动态地加载类,任意一个类的类名如果被作为参数传递给这个方法都将导致该类被加载到 JVM 内存中,如果这个类在类路径中没有被找到,那么此时就会在运行时抛出 ClassNotFoundException 异常。
解决方法:
解决该问题需要确保所需的类连同它依赖的包存在于类路径中,常见问题在于类名书写错误。
异常处理的两个基本原则:
- 尽量不要捕获类似 Exception 这样的通用异常,应该捕获特定异常。
- 不要生吞(swallow)异常。因为这样很可能导致非常难以诊断的异常情况。生吞异常往往是基于假设这段代码可能不会发生,但是程序如果后续以不可控的方式结束,此时会造成无法判断异常的具体位置,以及产生异常的具体原因。
从性能角度审视 Java 的异常处理机制:
- try-catch 代码段会产生额外的性能开销,即影响 JVN 对代码进行优化,所及建议【仅捕获有必要的代码段】,并且能就地解决,尽量不要一个大的 try 包住整段代码;利用异常控制程序流程远比条件语句(if / else, swittch)要低效。
- Java 每实例化一个 Exception,都会对当时的栈进行快照,这是一个相对比较重的操作。如果发生的非常频繁,这个开销就不能被忽略。
最后:Throw early, catch late 原则很重要。