面试题 Java 异常

说一下Java中的异常体系?

异常 - 图1

Error和Exception的区别?

Error(错误)

系统中的错误,是在程序编译时出现的错误,只能通过修改程序才能修正。一般是指与虚拟机相关的问题,如系统崩溃,虚拟机错误,内存空间不足,方法调用栈溢等。

Exception(异常)

表示程序可以处理的异常,可以捕获且可能恢复。遇到这类异常,应该尽可能处理异常,使程序恢复运行,而不应该随意终止异常。

写出最常见的 5 个 RuntimeException?

(1)java.lang.NullPointerException 空指针异常;出现原因:调用了未经初始化的对象或者是不存在的对象。
(2)java.lang.ClassNotFoundException 指定的类找不到;出现原因:类的名称和路径加载错误;通常都是程序
试图通过字符串来加载某个类时可能引发异常。
(3)java.lang.NumberFormatException 字符串转换为数字异常;出现原因:字符型数据中包含非数字型字符。
(4)java.lang.IndexOutOfBoundsException 数组角标越界异常,常见于操作数组对象时发生。
(5)java.lang.IllegalArgumentException 方法传递参数错误。
(6)java.lang.ClassCastException 数据类型转换异常。

如何处理异常?

异常的处理方式有两种:

  • 自己处理。
  • 向上抛, 交给调用者处理。

具体的处理方式的选择原则:

  • 自己明确的知道如何处理的, 就要处理掉。
  • 不知道如何处理的, 就交给调⽤者处理。

    注:异常, 不能捕获了之后什么也不做。或者只是使⽤e.printStacktrac

try()里面有⼀个return语句, 那么后面的finally{}里面的代码会不会被执行?什么时候执行,return前还是return后?

如果try中有return语句, 那么finally中的代码还是会执⾏。
因为return表示的是要整个方法体返回, 所以,finally中的语句会在return之前执⾏。
但是return前执行的finally块内,对数据的修改效果对于引用类型和值类型会所不同:

  1. // 测试 修改值类型
  2. static int f() {
  3. int ret = 0;
  4. try {
  5. return ret; // 返回 0,finally内的修改效果不起作用
  6. } finally {
  7. ret++;
  8. System.out.println("finally执行");
  9. }
  10. }
  11. // 测试 修改引用类型
  12. static int[] f2(){
  13. int[] ret = new int[]{0};
  14. try {
  15. return ret; // 返回 [1],finally内的修改效果起了作用
  16. } finally {
  17. ret[0]++;
  18. System.out.println("finally执行");
  19. }
  20. }

调用下面的方法,得到的返回值是什么?

  1. public int getNum() {
  2. try {
  3. int a = 1 / 0;
  4. return 1;
  5. } catch (Exception e) {
  6. return 2;
  7. } finally {
  8. return 3;
  9. }
  10. }

返回值为3。
代码在走到第 3 行的时候遇到了一个 MathException,这时第四行的代码就不会执行了,代码直接跳转到 catch语句中,走到第 6 行的时候,异常机制有这么一个原则如果在 catch 中遇到了 return 或者异常等能使该函数终止的话那么有 finally 就必须先执行完 finally 代码块里面的代码然后再返回值。因此代码又跳到第 8 行,可惜第 8 行是一个return 语句,那么这个时候方法就结束了,因此第 6 行的返回结果就无法被真正返回。如果 finally 仅仅是处理了一个释放资源的操作,那么该道题最终返回的结果就是 2。因此上面返回值是 3。

throw 和 throws 的区别?

throw

  • throw 语句用在方法体内,表示抛出异常,由方法体内的语句处理。
  • throw 是具体向外抛出异常的动作,所以它抛出的是一个异常实例,执行 throw 一定是抛出了某种异常。

    throws

  • throws 语句是用在方法声明后面,表示如果抛出异常,由该方法的调用者来进行异常的处理。

  • throws 主要是声明这个方法会抛出某种类型的异常,让它的使用者要知道需要捕获的异常的类型。
  • throws 表示出现异常的一种可能性,并不一定会发生这种异常。