参考文章: https://snailclimb.top/JavaGuide/#/
在 Java 中,所有的异常都有一个共同的祖先java.lang包中的 Throwable类。Throwable: 有两个重要的子类:Exception(异常) 和 Error(错误) ,二者都是 Java 异常处理的重要子类,各自都包含大量子类:
Error(错误):
是程序无法处理的错误,表示运行应用程序中较严重问题。大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。例如,Java虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现 OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止。
这些错误表示故障发生于虚拟机自身、或者发生在虚拟机试图执行应用时,如Java虚拟机运行错误(Virtual MachineError)、类定义错误(NoClassDefFoundError)等。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在 Java中,错误通过Error的子类描述。
Exception(异常):
是程序本身可以处理的异常。Exception 类有一个重要的子类 RuntimeException。RuntimeException 异常由Java虚拟机抛出。NullPointerException(要访问的变量没有引用任何对象时,抛出该异常)、ArithmeticException(算术运算异常,一个整数除以0时,抛出该异常)和 ArrayIndexOutOfBoundsException (下标越界异常)。
其实,可以简单把异常分为运行时异常和非运行时异常,
非运行时异常是RuntimeException以外的异常,类型上都属于Exception类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常,一般情况下不自定义检查异常,简单来说,需要由编译器告诉你需要try…catch的异常基本都是非运行时的异常,比如io异常,文件找不到异常等等;
运行时异常是RuntimeException,这个一般是逻辑上可能会有bug产生的异常,由虚拟机进行抛出,程序中可以选择捕获处理,也可以不处理,程序应该从逻辑角度尽可能避免这类异常的发生,比如空指针,索引越界,文件找不到异常等等.
捕获异常
- try 块: 用于捕获异常。其后可接零个或多个catch块,如果没有catch块,则必须跟一个finally块。
- catch 块: 用于处理try捕获到的异常。
- finally 块: 无论是否捕获或处理异常,finally块里的语句都会被执行。当在try块或catch块中遇到return
语句时,finally语句块将在方法返回之前被执行。
在以下4种特殊情况下,finally块不会被执行:
- 在finally语句块第一行发生了异常。 因为在其他行,finally块还是会得到执行
- 在前面的代码中用了System.exit(int)已退出程序。 exit是带参函数 ;若该语句在异常语句之后,finally会执行
- 程序所在的线程死亡。
- 关闭CPU。
注意: 当try语句和finally语句中都有return语句时,在方法返回之前,finally语句的内容将被执行,并且finally语句的返回值将会覆盖原始的返回值。如下:
public static int f(int value) {
try {
return value * value;
} finally {
if (value == 2) {
return 0;
}
}
}
如果调用 f(2),返回值将是0,因为finally语句的返回值覆盖了try语句块的返回值。
自定义异常
直接上代码
package test;
public class MyException {
public static void main(String[] args) {
demo d = new demo ();
int a = 10;
int b = -1;
try {
int div = d.div (a,b);
System.out.println (div);
}catch (fuShuException e){
System.out.println ("除数为"+b);
System.out.println (e.getMessage ());
}
}
}
class fuShuException extends Exception{
fuShuException(String msg){
super(msg);
}
}
class demo {
public int div(int a, int b)throws fuShuException{
if( b<= 0){
throw new fuShuException("除数不能为负数");
}
return a/b;
}
}
自定义异常就直接继承Exception这个类,有需要就重写构造方法,然后在需要抛出异常的地方使用关键字throws 进行指定异常
- 自定义异常:
1. class 异常类名 extends Exception
{
public 异常类名(String msg)
{
super(msg);
}
}
- 标识可能抛出的异常:
throws 异常类名 - 捕获异常:
1. try{}
catch(异常类名 y){}
- 方法解释
2. getMessage() //输出异常的信息
printStackTrace() //打印堆痕迹