没有不出错的程序,优雅的处理错误,是程序可靠性的关键
Q1:请对比 Exception 和 Error
- Exception 和 Error 都是继承了 Throwable 类
- Exception 是程序正常运行中,可以预料的意外情况,可能并且应该被捕获,进行相应处理
- Error是不会恢复错误,在正常情况下,不大可能出现的情况。若出现 Error ,说明程序处于不可恢复状态,所以不便于也不需要捕获。常见的比如 OutOfMemoryError 之类,都是 Error 的子类。
- Exception 又分为可检查异常和不可检查异常,可检查异常在源代码里必须显式地进行捕获处理,这是编译期检查的一部分。不检查异常就是所谓的运行时异常,类似 NullPointerException、ArrayIndexOutOfBoundsException 之类,通常是可以编码避免的逻辑错误,具体根据需要来判断是否需要捕获,并不会在编译期强制要求。
Q2:运行时异常与一般异常有什么区别?
Q3:处理异常有哪些注意的细节:
- 尽量不要捕获类似 Exception 这样的通用异常,而是应该捕获特定异常
- 不要生吞(swallow)异常,要选用最合适的方式输出。
- 遵循Throw early, catch late
Q4:为什么不能异常控制逻辑:
- try-catch 代码段会产生额外的性能开销,或者换个角度说,它往往会影响 JVM 对代码进行优化,所以建议仅捕获有必要的代码段,尽量不要一个大的 try 包住整段的代码;与此同时,利用异常控制代码流程,也不是一个好主意,远比我们通常意义上的条件语句(if/else、switch)要低效。
- Java 每实例化一个 Exception,都会对当时的栈进行快照,这是一个相对比较重的操作。如果发生的非常频繁,这个开销可就不能被忽略了。
Q5:“NoClassDefFoundError 和 ClassNotFoundException 有什么区别
- NoClassDefFoundError是一个错误(Error),而ClassNOtFoundException是一个异常,在Java中对于错误和异常的处理是不同的,我们可以从异常中恢复程序但却不应该尝试从错误中恢复程序。
- 如果JVM或者ClassLoader实例尝试加载(可以通过正常的方法调用,也可能是使用new来创建新的对象)类的时候却找不到类的定义。要查找的类在编译的时候是存在的,运行的时候却找不到了。这个时候就会导致NoClassDefFoundError.