目录
概念:
异常是在程序中导致程序中断运行的一种指令流。
例如下面的操作代码:
public class Demo {
public static void main(String[] args) {
int i=20;
int j=0;
System.out.println("-----------开始计算-----------");
int temp= i/j;
System.out.println("temp="+temp);
}
}
运算结果为:
以上的代码在“int temp = i / j ;”位置处产生了异常,一旦产生异常之后,异常之后的语句将不再执行了,所以现在的程序并没有正确的执行完毕之后就退出了。那么,为了保证程序出现异常之后仍然可以正确的执行完毕,所以要采用异常的处理机制。
异常体系结构
异常指的是Exception类。在Java中存在一个父类Throwable,Throwable有两个子类:
- Error:表示 错误,指的是JVM发出的错误操作,这种只能尽量避免,无法用代码处理。
- Exception: 一般表示程序中的错误,一般用try..catch进行处理。
异常的分类
细心的同学可能发现了,这里分受检异常和非受检异常。
所谓的非受检异常通常时由编程错误导致的,所以在编写程序时,并不要求必须使用异常处理机制来处理这类异常。
而受检异常在编译时就必须做出处理,否则无法通过编译。
处理异常
如果要想对异常进行处理,则必须采用标准的处理格式。
处理方式有两种:
- 抛出异常(throws):在执行一个方法时,如果发生异常,则这个方法生成代表该异常的一个对象,停止当前执行路径,并把异常对象提交给JVM。
- 捕获异常(try...catch):JVM得到该异常后,寻找相应的代码来处理该异常。JVM在方法的调用栈中查找,从生成异常的方法开始回溯,直到找到相应的异常处理代码为止。
捕获异常
try{
// 有可能发生异常的代码段
}catch(异常类型1 对象名1){
// 异常的处理操作
}catch(异常类型2 对象名2){
// 异常的处理操作
} ...
finally{
// 异常的统一出口
}
多异常捕获写法:
catch(异常类型1 |异常类型2 对象名){
//表示此块用于处理异常类型1 和 异常类型2 的异常信息
}
多异常捕获的注意点:
1、捕获更大的异常不能放在捕获更小的异常之前。
2、如果为了方便,则可以将所有的异常都使用Exception进行捕获。
try+catch+finally的处理流程
- 一旦产生异常,则JVM会自动产生一个异常类的实例化对象--包含异常信息。
- 那么,此时如果异常发生在try语句,则会自动找到匹配的catch语句执行,如果没有在try语句中,则会将异常抛出。
- 所有的catch根据方法的参数匹配异常类的实例化对象,如果匹配成功,则表示由此catch进行处理。
- 如果没有catch匹配得上,main也未处理,则自动将异常抛给main的调用者JVM
- JVM对异常信息进行了响应,将异常信息显示在控制台,中断处理。
- 在进行异常处理之后,在异常的处理格式中还有一个finally语句,那么此语句将作为异常的统一出口,不管是否产生了异常,最终都要执行finally里面的代码。
新增“try-with-resource"
JDK7之后,新增了“try-with-resource"。它可以自动关闭实现了AutoClosable接口的类,实现类需要close()方法。在编译时仍然会进行转化成try-catch-finally语句。
public class Demo { public static void main(String[] args) { try (FileReader reader=new FileReader("d:/a.txt")){ //其他语句 } catch (Exception e) { e.printStackTrace(); } } }
抛出异常
随异常一起的还有一个称为throws关键字,此关键字主要在方法的声明上使用,表示方法中不处理异常,而交给调用处处理。
格式:
返回值 方法名称() throws Exception{
}
throw 关键字表示在程序中人为的抛出一个异常,因为从异常处理机制来看,所有的异常一旦产生之后,实际上抛出的就是要给异常类的实例化对象,那么此对象也可以由throw直接抛出。
例子: throw new Exception("抛着玩");
自定义异常(了解)
编写一个类, 继承Exception,并重写一参构造方法 即可完成自定义受检异常类型。
编写一个类, 继承RuntimeExcepion,并重写一参构造方法 即可完成自定义运行时异常类型。
例如:
class MyException extends Exception{ // 继承Exception,表示一个自定义异常类
public MyException(String msg){
super(msg) ; // 调用Exception中有一个参数的构造
}
};
自定义异常可以做很多事情, 例如:
class MyException extends Exception{
public MyException(String msg){
super(msg) ;
//在这里给维护人员发短信或邮件, 告知程序出现了BUG。
}
};
几个常见的问题:
1. try-catch-finally 中哪个部分可以省略?
答: catch和finally可以省略其中一个 , catch和finally不能同时省略。
同时,finally为异常处理提供一个统一的出口,所以最多只能存在一个finally2. try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?
答:finally中的代码会执行
执行流程:
- 先计算返回值, 并将返回值存储起来, 等待返回。
- 执行finally代码块。
- 将之前存储的返回值, 返回出去。
需注意的是:
- 返回值是在finally运算之前就确定了,并且缓存了,不管finally对该值做任何的改变,返回的值都不会改变
- finally代码中不建议包含return,因为程序会在上述的流程中提前退出,也就是说返回的值不是try或catch中的值
- 如果在try或catch中停止了JVM,则finally不会执行,例如停电。 或通过如下代码退出JVM:System.exit(0);