文章目录
- 前言
- 一、try catch finally语句的作用
- 二、try catch finally语句的执行顺序
- 三、try catch finally语句的return问题
- 四、try catch finally语句的升级版
- 确保资源一定被关闭
- try-with-Resource语句
- 总结
前言
今天来整理Java基础知识点,try catch finally的执行顺序问题
一、try catch finally语句的作用
众所周知,我们在学习Java异常处理时,都会接触到try catch finally这三个关键字,他们三个的出场大致是如下的顺序:
1.
try {
//逻辑代码
}catch(exception e){
//异常处理代码
} finally{
//一定要执行的代码
}
2.
try {
//逻辑代码
}catch(exception e){
//异常处理代码
}
3.
try{
//逻辑代码
}finally{
//一定要执行的代码
}
try语句负责存放程序逻辑代码,catch负责存放处理程序中抛出的异常;这二者存在的意义显而易见,前者负责存放程序正常必须要完成的代码逻辑,而当try中的代码抛出一个异常时,就必须要有代码去捕获这个异常,去处理代码逻辑,因为有了catch中存放的语句。
而正常情况下,当try中的语句请求本地已知的一些资源时,当程序发生异常并返回时,如果try中的方法已经请求到了自己已知的一些本地资源,就必须要对这些资源进行清理,否则当垃圾回收机制不能及时对这部分资源进行回收时,便会产生内存泄漏等问题。
因而finally中的语句其实扮演的角色便是在程序抛出异常后,依然对这些已获得的资源进行清理,完成扫尾工作的角色。
二、try catch finally语句的执行顺序
来看以下一段伪码
var in = new FileInputStream(...);
try{
//1
code statement
//2
}
catch (IOException e){
//3
show error message
//4
}
finally{
//5
in.close();
}
//6
结合以上伪码进行分析,try catch finally的执行顺序大致有以下几种情况
- try中代码运行无异常,则按正常顺序执行 1-2-5-6
- try中代码发生异常,catch 中的代码没有抛出异常,则执行顺序位1-3-4-5-6
- try中代码发生异常,catch 中的代码抛出异常,即将这个异常扔回给调用者本身 ,不会执行try-catch-finally之后的代码则执行顺序位1-3-5
- try中代码发生异常,catch 中没有任意可以捕获对应异常类型的语句,则将这个异常扔回给调用者本身,执行顺序 1-5
从上可以看到,finally代码无论如何都会与运行,扮演着最后收尾的扫地工作
三、try catch finally语句的return问题
面试中经常会问大家 try-catch-finally 语句里面有return的情况下,最终的return结果是啥;
其实官方是十分不建议把改变控制流的语句放到finally中,因为这可能导致覆盖错误
来看以下一段伪码
public static int parseInt(String s){
try{
return Integer.parseInt(s);
}catch(NumberFormatException e){
e.printStachTrace();
}finally{
return 2;
}
return 3;
}
我们在IDEA中执行查看结果
发现IDEA直接要我取出return 3;好家伙
执行结果如下:
我们很容易发现finally中的return会对try中的return进行覆盖;
甚至说,如果输入不合法,try中的方法抛出一个异常,finally中的方法甚至会直接覆盖掉这个异常。
四、try catch finally语句的升级版
确保资源一定被关闭
试想,在上文中的写法中,如果finally中的语句in.close()发生异常,那么又该交给谁来处理呢;如果try语句知识为了确保资源一定被关闭,我们不妨提供以下做法:
var in = new FileInputStream(...);
try{
try{
//1
code statement
//2
}
finally{
io.close();
}
}
catch (IOException e){
//3
show error message
//4
}
这种做法更清楚,功能性也更强,内层try确保资源被关闭,外层try确保报告发生的错误
try-with-Resource语句
如果仅仅是为了获取资源,又觉得finally语句过于繁琐的,不妨采用以下结构
try (Resource res=...){
work with res
}
当try块退出时,会自动调用res.close();
例:
try (var in new Scanner(new FileInputStream(...)) ){
while(in.hasNext())
out.println(in.next().toUpperCase())
}
总结
以上是try catch finally语句的复习,关于这一部分新的知识点想补充的可以留言笔者,感谢。