关于try-catch-finally中的return
分为以下几种情况:
举例:
情况1:try{} catch(){}finally{} return;
显然程序按顺序执行。
情况2:try{ return; }catch(){} finally{} return;
程序执行try块中return之前(包括return语句中的表达式运算)代码;
再执行finally块,最后执行try中return;
finally块之后的语句return,因为程序在try中已经return所以不再执行。
public static int TryCF(){
int x = 1;
try{
x = x + 1;
System.out.println("try: x = " + x);
return x+1;
}
catch(Exception e){
System.out.println("catch: x = " + x);
return x+2;
}
finally{
x = x + 3;
System.out.println("finally: x = " + x);
}
// return x+3; //出错需要注释掉
}
输出:
try: x = 2
finally: x = 5
x的最终值:3 注意:return当前块中的数据3,而不是最终的x值5
情况3:try{ } catch(){return;} finally{} return;
程序先执行try,如果遇到异常执行catch块,
有异常:则执行catch中return之前(包括return语句中的表达式运算)代码,再执行finally语句中全部代码,最后执行catch块中return. finally之后代码不再执行。
无异常:执行完try再finally再return.
public static int TryCF(){
int x = 1;
try{
x = x + 1;
x = 2/0;
System.out.println("try: x = " + x);
}
catch(Exception e){
System.out.println("catch: x = " + x);
return x+2;
}
finally{
x = x + 3;
System.out.println("finally: x = " + x);
}
return x+3;
}
输出:
catch: x = 2
finally: x = 5
x的最终值:4
情况4:try{ return; }catch(){} finally{return;}
程序执行try块中return之前(包括return语句中的表达式运算)代码;
再执行finally块,因为finally块中有return所以提前退出。
code:
public static int TryCF(){
int x = 1;
try{
x = x + 1;
System.out.println("try: x = " + x);
return x+1;
}
catch(Exception e){
System.out.println("catch: x = " + x);
return x+2;
}
finally{
x = x + 3;
System.out.println("finally: x = " + x);
return x+3;
}
输出:
try: x = 2
finally: x = 5
x的最终值:8
情况5:try{} catch(){return;}finally{return;}
程序执行catch块中return之前(包括return语句中的表达式运算)代码;
再执行finally块,因为finally块中有return所以提前退出。
public static int TryCF(){
int x = 1;
try{
x = x + 1;
x = 2/0;
System.out.println("try: x = " + x);
}
catch(Exception e){
System.out.println("catch: x = " + x);
return x+2;
}
finally{
x = x + 3;
System.out.println("finally: x = " + x);
return x+3;
}
}
输出:
catch: x = 2
finally: x = 5
x的最终值:8
情况6:try{ return;}catch(){return;} finally{return;}
程序执行try块中return之前(包括return语句中的表达式运算)代码;
有异常:执行catch块中return之前(包括return语句中的表达式运算)代码;
则再执行finally块,因为finally块中有return所以提前退出。
无异常:则再执行finally块,因为finally块中有return所以提前退出。
有异常:
public static int TryCF(){
int x = 1;
try{
x = x + 1;
x = 2/0;
System.out.println("try: x = " + x);
return x+1;
}
catch(Exception e){
System.out.println("catch: x = " + x);
return x+2;
}
finally{
x = x + 3;
System.out.println("finally: x = " + x);
return x+3;
}
}
输出:
catch: x = 2
finally: x = 5
x的最终值:8
无异常:
public static int TryCF(){
int x = 1;
try{
x = x + 1;
System.out.println("try: x = " + x);
return x+1;
}
catch(Exception e){
System.out.println("catch: x = " + x);
return x+2;
}
finally{
x = x + 3;
System.out.println("finally: x = " + x);
return x+3;
}
}
输出:
try: x = 2
finally: x = 5
x的最终值:8
对于情况3绿了有如下解释:
因为java都是按值传递的。return的若是对象,则先把对象的副本保存起来,也就是说保存的是指向对象的地址。若对原来的对象进行修改。对象的地址仍然不变,return的副本仍然是指向这个对象,所用finally中对对象的修改仍然有作用。而基本数据类型保存的是原原本本的数据,return保存副本后,在finally中修改都是修改原来的数据。副本中的数据还是不变,所以finally中修改对return无影响。
Code:
private static List<string> test1()
{
List<string> strlist = new List<string>();
strlist.Add("zs");
strlist.Add("ls");
strlist.Add("ww");
strlist.Add("mz");
try
{
strlist.Add("wq");
return strlist;
}
finally
{
strlist.Add("yyy");
}
}
输出:
zs
ls
ww
mz
wq
yyy
关于返回值是否改变总结:
1、如果finally中没有return语句,但是改变了要返回的值,这里有点类似与引用传递和值传递的区别,分以下两种情况:
1)如果return的数据是基本数据类型或文本字符串,则在finally中对该基本数据的改变不起作用,try中的return语句依然会返回进入finally块之前保留的值。
2)如果return的数据是引用数据类型,而在finally中对该引用数据类型的属性值的改变起作用,try中的return语句返回的就是在finally中改变后的该属性的值。
2、如果finally中有return 语句,就提前退出,返回的也是finally中的最终值(无论数据类型)
结论:
1、不管是否出现异常,finally块中代码都会执行;
2、当try和catch中有return时,finally仍然会执行;
3、finally是在return后面的表达式运算后执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的;
4、finally中最好不要包含return,否则程序会提前退出,返回值不是try或catch中保存的返回值。
5、若finally无return,在try catch中有return,则执行完finally,直接返回去return ,不在执行fianlly以后的语句。
6、任何执行try 或者catch中的return语句之前,都会先执行finally语句,如果finally存在的话。如果finally中有return语句,那么程序就return了,所以finally中的return是一定会被return的,编译器把finally中的return实现为一个warning提前退出。在哪个块中最终return 的值是本块的局部最终值(见绿)
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· .NET10 - 预览版1新功能体验(一)