Throwable类是Java语言所有错误或异常的超类(两个直接子类:Error和Exception)
Java异常处理类的关系图Error vs Exception
Java程序在执行过程中所发生的异常事件可分为两类:
Error:Error类对象由Java虚拟机生成并抛出,大多数错误与代码编写者所执行的操作无关。例如,Java虚拟机运行错误(Virtual MachineError),当JVM不再有继续执行操作所需的内存资源时,将出现OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止;还有发生在虚拟机试图执行应用时,如类定义错误(NoClassDefFoundError)、链接错误(LinkageError)。这些错误是不可查的,因为它们在应用程序的控制和处理能力之外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在Java中,错误通常是使用Error的子类描述。
Exception:在Exception分支中有一个重要的子类RuntimeException(运行时异常),该类型的异常自动为你所编写的程序定义ArrayIndexOutOfBoundsException(数组下标越界)、NullPointerException(空指针异常)、ArithmeticException(算术异常)、MissingResourceException(丢失资源)、ClassNotFoundException(找不到类)等异常,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生;而RuntimeException之外的异常我们统称为非运行时异常,类型上属于Exception类及其子类,从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常,一般情况下不自定义检查异常。
Error和Exception的区别:Error通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时,Java虚拟机(JVM)一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。
检查异常vs不受检查异常
检查异常:在正确的程序运行过程中,很容易出现的、情理可容的异常状况,在一定程度上这种异常的发生是可以预测的,并且一旦发生该种异常,就必须采取某种方式进行处理。
除了RuntimeException及其子类以外,其他的Exception类及其子类都属于检查异常,当程序中可能出现这类异常,要么使用try-catch语句进行捕获,要么用throws子句抛出,否则编译无法通过。
不受检查异常:包括RuntimeException及其子类和Error。
不受检查异常为编译器不要求强制处理的异常,检查异常则是编译器要求必须处置的异常。
异常处理关键字
Java异常处理涉及到五个关键字,分别是:try、catch、finally、throw、throws。
- try: 用于监听。将要被监听的代码(可能抛出异常的代码)放在try语句块之内,当try语句块内发生异常时,异常就被抛出。
- catch:用于捕获异常。catch用来捕获try语句块中发生的异常。
—finally:finally语句块总是会被执行。它主要用于回收在try块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。只有finally块,执行完成之后,才会回来执行try或者catch块中的return或者throw语句,如果finally中使用了return或者throw等终止方法的语句,则就不会跳回执行,直接停止。 - throw:用于抛出异常。
- throws:用在方法签名中,用于声明该方法可能抛出的异常。
注意:try后面如果跟多个catch,那么小范围的异常放前面,大范围的异常放后面(根据多态的原理,如果大的放前面,就会将所有的子类对象接收,后面的catch就没有意义了)
常见问题
final、finally、finalize的区别
- final 可以用来修饰类、方法、变量,分别有不同的意义,final 修饰的 class 代表不可以继承扩展,final的变量是不可以修改的,而final的方法也是不可以重写的(override)。
- finally 则是Java保证重点代码一定要被执行的一种机制。我们可以使用 try-finally 或者 try-catch-finally 来进行类似关闭 JDBC 连接、保证 unlock 锁等动作。
- finalize是基础类 java.lang.Object的一个方法,它的设计目的是保证对象在被垃圾收集前完成特定资源的回收。finalize 机制现在已经不推荐使用,并且在 JDK 9 开始被标记为 deprecated。
需要关闭的连接等资源时,相比finally,更推荐使用Java 7中添加的try-with-resources语句,因为通常Java平台能够更好地处理异常情况,编码量也要少很多。如果确实需要额外处理,可以考虑 Java 提供的 Cleaner 机制或者其他替代方法。
如果在执行到finally之前JVM退出了,比如System.exit(0)),则finally代码不会执行
如果catch里面有return语句,finally里面的代码还会执行吗?
代码示例:
class FinallyDemo {
public static void main(String[] args) {
System.out.println(getInt());
}
public static int getInt() {
int a = 10;
try {
System.out.println(a / 0);
a = 20;
} catch (ArithmeticException e) {
a = 30;
return a;
} finally {
a = 40;
return a+1;
}
// return a;
}
}
上面的实验代码可以解答所有问题。
- 如果finally没有return的话,catch在运行return之前将会运行finally的逻辑,然后再跳转回catch的return,输出30。
- 如果finally如果有return的话,catch在运行return之前会运行finally的逻辑,然后直接运行finally的return,输出41。
异常注意事项
- 子类不能抛出父类没有的异常。子类抛出的的异常与父类抛出的异常相同或者是父类抛出的异常的子类
- 如果父类中被重写的方法没有异常抛出,那么子类的方法绝对不可以抛出异常。如果子类中有异常发生,那么子类只能try,不能throws
网友评论