文章目录

  • 前言
  • 一、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中执行查看结果

java try inputstream 被自动close java try catch finally_执行顺序


发现IDEA直接要我取出return 3;好家伙

执行结果如下:

java try inputstream 被自动close java try catch finally_开发语言_02


我们很容易发现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语句的复习,关于这一部分新的知识点想补充的可以留言笔者,感谢。