异常处理
什么是异常处理?
异常是在程序中导致程序中断运行的一种指令流。
例如,现在有如下的操作代码:
public class ExceptionDemo01{
public static void main(String argsp[]){
int i = 10 ; int j = 0 ;
System.out.println("============= 计算开始 =============") ;
int temp = i / j ; // 进行除法运算
System.out.println("temp = " + temp) ;
System.out.println("============= 计算结束 =============") ;
}
};
运行结果:
============= 计算开始 =============
Exception in thread "main" java.lang.ArithmeticException: / by zero
at ExceptionDemo01.main(ExceptionDemo01.java:6)
以上的代码在 “int temp = i / j ;” 位置处产生了异常,一旦产生异常之后,异常之后的语句将不再执行了,所以现 在的程序并没有正确的执行完毕之后就退出了。 那么,为了保证程序出现异常之后仍然可以正确的执行完毕,所以要采用异常的处理机制。
异常以及错误 (Exception and Error)
Error: 这是无法通过程序来弥补的错误,通常跟硬件或者网络相关,这类错误我们通常在编程中不考虑
Exception: 这种是我们在编程中需要考虑的异常。通常分为受检异常和非受检异常。顾名思义就是一种是在run之前就可以检测出来,一般由编译器负责。一种是运行之后才知道,比如除0异常,空指针,越界之类的。
处理异常
如果要想对异常进行处理,则必须采用标准的处理格式,处理格式语法如下:
try{
// 有可能发生异常的代码段
}catch(异常类型1 对象名1){
// 异常的处理操作
}catch(异常类型2 对象名2){
// 异常的处理操作
} ...
finally{
// 异常的统一出口
}
处理逻辑是:
1、 一旦产生异常,则系统会自动产生一个异常类的实例化对象。
2、 那么,此时如果异常发生在try语句,则会自动找到匹配的catch语句执行,如果没有在try语句中,则会将异 常抛出.
3、 所有的catch根据方法的参数匹配异常类的实例化对象,如果匹配成功,则表示由此catch进行处理。
在Java中,如果发生异常,如果没做处理,会从出现异常的地方返回调用之前,直到JVM。比如函数A调用函数B,B不做处理的话,B异常会返回A看A有没有做异常处理,A也没的话就会继续返回… 最后会返回给JVM,它会中断程序并对异常信息进行响应,输出到终端。
finally (重点讨论)
finally语句是不管什么情况下都一定会执行,也就是说不管进入try还是进入catch还是都不进入,除非程序崩溃或者是从内存中清除或者是遭遇关机等不可抗力因素。同时finally也很有趣,因为即使前面try catch return了也一样会执行。
来看下代码:
在这里进入了try模块,但是结果返回的其实是p.age = 28。这是为什么呢?
这里就要讲一下return的机制。return的机制是先复制一个p,然后做返回准备。在准备之后返回之前,会执行finally。因为这里复制的p是个对象,所以复制的是地址。那么在finally里,把这个地址里的age属性给修改了,那么返回以后打印出来的age自然就是28。画成内存来看就是如下图(小的是栈,大的是堆):这里备份的是地址。
那这里其实聪明的人就会举一反三,如果返回的是一个基础类型呢?比如int。
在这种情况里基础类型是直接备份值的,所以根据执行顺序,会先备份值 = a,然后执行finally,然后返回。这时候返回值就是原来的10。