异常处理
什么是异常?
异常是在程序中导致程序中断运行的一种指令流。
一旦产生异常之后,异常之后的语句将不再执行了,就会导致程序在并没有正确的执行完毕就退出了。所以,为了保证程序出现异常之后仍然可以正确的执行完毕,所以要采用异常的处理机制。
异常的产生到程序中断的过程:
如何处理异常?
如果要想对异常进行处理,则必须采用标准的处理格式,处理格式语法如下:
try{
// 有可能发生异常的代码段
}catch(异常类型1 对象名1){
// 异常的处理操作
}catch(异常类型2 对象名2){
// 异常的处理操作
} ...
finally{
// 异常的统一出口
}
try+catch的处理流程:
- 一旦产生异常,则系统会自动产生一个异常类的实例化对象。
- 那么,此时如果异常发生在try语句,则会自动找到匹配的catch语句执行,如果没有在try语句中,则会将异常抛出。
- 所有的catch根据方法的参数匹配异常类的实例化对象,如果匹配成功,则表示由此catch进行处理。
你好! 这是你第一次使用 Markdown编辑器 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章,了解一下Markdown的基本语法知识。
finally:
在进行异常的处理之后,在异常的处理格式中还有一个finally语句,那么此语句将作为异常的统一出口,不管是否产生了异常,最终都要执行此段代码。
异常体系结构:
异常指的是Exception,Exception类在Java中存在一个父类Throwable(可能的抛出)
Throwable存在两个子类:
- Error:表示的是错误,是JVM发出的错误操作,只能尽量避免,无法用代码处理。
- Exception:一般表示所有程序中的错误,所以一般在程序中将进行try…catch的处理。
其中我们需要关注的是Exception,Exception又分为受检异常和非受检异常:
- 受检异常:在编译时被强制检查的异常称为"受检查的异常"。即在方法的声明中声明的异常。
- 非受检异常:在方法的声明中没有声明,但在方法的运行过程中发生的各种异常被称为"不被检查的异常"。这种异常是错误,会被自动捕获。
多异常捕获的注意点:
- 捕获更粗的异常不能放在捕获更细的异常之前。
- 如果为了方便,则可以将所有的异常都使用Exception进行捕获。
特殊的多异常捕获写法(不常用):
catch(异常类型1 |异常类型2 对象名){
//表示此块用于处理异常类型1 和 异常类型2 的异常信息
}
throws关键字
此关键字主要在方法的声明上使用,表示方法中不处理异常,而交给调用处处理。 当你写了一个方法,这个方法会因为传入的参数出现异常,就可以选择抛出异常。
格式:
返回值 方法名称()throws Exception{
}
throw关键字
throw关键字表示在程序中人为的抛出一个异常,因为从异常处理机制来看,所有的异常一旦产生之后,实际上抛出的就是一个异常类的实例化对象,那么此对象也可以由throw直接抛出。
代码:
throw new Exception("XXXX") ;
RuntimeExcepion与Exception的区别
注意观察如下方法的源码:
public static int parseInt(String text)throws NumberFormatException
此方法抛出了异常, 但是使用时却不需要进行try catch捕获处理,原因:
因为NumberFormatException并不是Exception的直接子类,而是RuntimeException的子类,只要是RuntimeException的子类,则表示程序在操作的时候可以不必使用try…catch进行处理,如果有异常发生,则由JVM进行处理。当然,也可以通过try catch处理。
自定义异常类(了解即可)
编写一个类, 继承Exception,并重写一参构造方法即可完成自定义受检异常类型。
编写一个类, 继承RuntimeExcepion,并重写一参构造方法即可完成自定义运行时异常类型。
自定义异常类的应用,例如:
class MyException extends Exception{
public MyException(String msg){
super(msg) ;
//在这里给维护人员发短信或邮件, 告知程序出现了BUG。
}
};
异常处理常见面试题
- try-catch-finally 中哪个部分可以省略?
答: catch和finally可以省略其中一个 , catch和finally不能同时省略
注意:格式上允许省略catch块, 但是发生异常时就不会捕获异常了,我们在开发中也不会这样去写代码。 - try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?
答:finally中的代码会执行
执行流程:- 先计算返回值,并将返回值存储起来,等待返回
- 执行finally代码块
- 将之前存储的返回值,返回出去
需注意:- 返回值是在finally运算之前就确定了,并且缓存了,不管finally对该值做任何的改变,返回的值都不会改变
- finally代码中不建议包含return,因为程序会在上述的流程中提前退出,也就是说返回的值不是try或 catch中的值
- 如果在try或catch中停止了JVM,则finally不会执行.例如停电- -, 或通过如下代码退出 JVM:System.exit(0);(在XX情况下finally是否会执行?)
案例1:
public static int get(){
int a = 10;
try{
return a;
}catch(Exception e){
}
Finally{
a = 20;
}
}
//此时a的返回值为10
案例2:
public static Person get(){
Person p = new Person();
p.age =18;
try{
return p;
}catch(Exception e){
}
Finally{
p.age = 28;
}
}
//此时调用p.age输出的是28
以上案例中finally会执行的原因是在执行return语句时,它不会直接返回值,而是先备份一份模板,finally就是在这时候执行的,当finally执行完后,return将备份的模板返回。案例1中备份的是10,所以最后返回的还是10,而案例2中备份的是对象的地址,指向的是同一个对象,所以返回后输出的值会是finally中改变的值。