异常
定义:所谓异常指的就是程序在 运行时 出现错误时通知调用者的一种机制。
一些常见的异常:
数组下标越界
int[] arr = {1, 2, 3};
System.out.println(arr[100]);
// 执行结果
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 100
访问null对象
public class Test {
public int num = 10;
public static void main(String[] args) {
Test t = null;
System.out.println(t.num);
}
}
// 执行结果
Exception in thread "main"java.lang.NullPointerException
关于运行时、编译时
有些错误是这样的,例如将 System.out.println 拼写错了,写成system.out.println。此时编译过程中就会出错,这是 “编译期” 出错。
而运行时指的是程序已经编译通过得到 class 文件了, 再由 JVM 执行过程中出现的错误。
异常的基本用法:
基本语法:
try{
有可能出现异常的语句 ;
}catch (异常类型 异常对象) {
...
}
finally {
异常的出口
}
- try 代码块中放的是可能出现异常的代码。
- catch 代码块中放的是出现异常后的处理行为。
- finally代码块中的代码用于处理善后工作, 会在最后执行。
- 其中 catch 和 finally 都可以根据情况选择加或者不加。
- 我们发现, 一旦 try 中出现异常, 那么 try 代码块中的程序就不会继续执行, 而是交给 catch 中的代码来执行. catch 执行完毕会继续往下执行.
- 一段代码可能会抛出多种不同的异常,不同的异常有不同的处理方式。因此可以搭配多个 catch 代码块。
- finally 表示最后的善后工作,例如释放资源。无论是否存在异常,finally 中的代码一定都会执行到。
- Exception 类是所有异常类的父类。因此可以用这个类型表示捕捉所有异常。(不推荐使用)
如果多个异常的处理方式是完全相同, 也可以写成这样:
catch (ArrayIndexOutOfBoundsException | NullPointerException e) {
...
}
关于调用栈
方法之间是存在相互调用关系的, 这种调用关系我们可以用 “调用栈” 来描述。在 JVM 中有一块内存空间称为 “虚拟机栈” 专门存储方法之间的调用关系, 当代码中出现异常的时候,我们就可以使用 e.printStackTrace(); 的方式查看出现异常代码的调用栈。
异常处理流程
- 程序先执行 try 中的代码
- 如果 try 中的代码出现异常,就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配。
- 如果找到匹配的异常类型, 就会执行 catch 中的代码 。
- 如果没有找到匹配的异常类型,就会将异常向上传递到上层调用者。
- 无论是否找到匹配的异常类型,finally 中的代码都会被执行到(在该方法结束之前执行)。
- 如果上层调用者也没有处理的了异常,就继续向上传递。
- 一直到 main 方法也没有合适的代码处理异常,就会交给 JVM 来进行处理, 此时程序就会异常终止。
抛出异常
除了 Java 内置的类会抛出一些异常之外, 程序猿也可以手动抛出某个异常。使用 throw 关键字完成这个操作.
public static void main(String[] args) {
System.out.println(divide(10, 0));
}
public static int divide(int x, int y) {
if (y == 0) {
throw new ArithmeticException("抛出除 0 异常");
}
return x / y;
}
// 执行结果
Exception in thread "main" java.lang.ArithmeticException: 抛出除 0 异常
at demo02.Test.divide(Test.java:14)
at demo02.Test.main(Test.java:9)
我们可以使用 throws 关键字,把可能抛出的异常显式的标注在方法定义的位置。从而提醒调用者要注意捕获这些异常。
public static int divide(int x, int y) throws ArithmeticException {
if (y == 0) {
throw new ArithmeticException("抛出除 0 异常");
}
return x / y;
}
java异常体系
下图表示 Java 内置的异常类之间的继承关系:
- 顶层类 Throwable 派生出两个重要的子类, Error 和 Exception
- 其中 Error 指的是 Java 运行时内部错误和资源耗尽错误. 应用程序不抛出此类异常. 这种内部错误一旦出现, 除了告知用户并使程序终止之外, 再无能无力.这种情况很少出现.
- Exception 是我们程序猿所使用的异常类的父类。
- 其中 Exception 有一个子类称为RuntimeException ,这里面又派生出很多我们常见的异常NullPointerException,IndexOutOfBoundsException 等.
Java语言规范将派生于 Error 类或 RuntimeException 类的所有异常称为 非受查异常,所有的其他异常称为 受查异常。
受查异常。如果不显式处理,编译无法通过
显示处理的两种方式:
- 使用 try catch 包裹起来
- 在方法上加上异常说明, 相当于将处理动作交给上级调用者