finally块中的代码什么时候被执行?finally是不是一定会被执行?(见最后引申)

本文探讨了Java中try-catch-finally语句中的return语句执行时机及finally块的作用,通过实例展示了不同情况下finally块内return的影响,并讨论了finally块执行的条件。

问题描述:try{}里有一个return语句,那么紧跟在这个try{}后面的finally{}中的代码是否会被执行?如果会的话,什么时候被执行,在return之前还是return之后?


在Java语言的异常处理中,finally块的作用就是为了保证无论出现什么情况,finally块里的代码一定会被执行。由于程序执行return就意味着结束对当前函数的调用并跳出这个函数体,因此任何语句要执行都只能在return前执行(除非碰到exit函数),因此finally块里的代码也是在return之前执行的。此外,如果try-finally或者catch-finally中都有return,那么finally块中的return将会覆盖别处的return语句,最终返回到调用者那里的是finally中return的值。下面通过一个例子来说明这个问题:

package com.js;
/**
 * try-catch中有return语句,finally中代码运行时机问题
 * @author jiangshuai
 */

public class Test{
	public static int testFinally(){
		try {
			return 1;
		} catch (Exception e) {
			return 0;
		}finally{
			System.out.println("execute finally");
		}
	}
	public static void main(String[] args){
		int result = testFinally();
		System.out.println(result);
	}
}
运行结果:

execute finally
1

从上面这个例子中可以看出,在执行return语句前确实执行了finally块中的代码。紧接着,在finally块里放置个return语句,来看看到底最终返回的是哪个return语句的值,例子如下图所示:

package com.js;
/**
 * try-catch中有多个return语句,研究return的是哪一个
 * @author jiangshuai
 */

public class Test{
	public static int testFinally(){
		try {
			return 1;
		} catch (Exception e) {
			return 0;
		}finally{
			System.out.println("execute finally");
			return 3;
		}
	}
	public static void main(String[] args){
		int result = testFinally();
		System.out.println(result);
	}
}
运行结果:

execute finally
3

从以上运行结果可以看出,当finally块中有return语句时,将会覆盖函数中其他return语句。此外,由于在一个方法内部定义的变量都存储在栈中,当这个函数结束后,其对应的栈就会被回收,此时在其方法体中定义的变量将不存在了,因此,对基本类型的数据,在finally块中改变return的值对返回值没有任何影响,而对引用类型的数据会有影响(详见 

Java中的值传递与引用传递详解

)。下面通过一个例子来说明这个问题:

package com.js;
/**
 * 在finally块中改变基本数据类型、引用类型对比
 * @author jiangshuai
 */

public class Test{
	public static int testFinally1(){
		int result = 1;
		try {
			result = 2;
			return result;
		} catch (Exception e) {
			return 0;
		}finally{
			result = 3;
			System.out.println("execute finally1");
		}
	}
	public static StringBuffer testFinally2(){
		StringBuffer s = new StringBuffer("Hello");
		try {
			return s;
		} catch (Exception e) {
			return null;
		}finally{
			s.append(" World");
			System.out.println("execute finally2");
		}
	}
	public static void main(String[] args){
		int result = testFinally1();
		System.out.println(result);
		StringBuffer resultRef = testFinally2();
		System.out.println(resultRef);
	}
}
运行结果:

execute finally1
2
execute finally2
Hello World

程序在执行到return时会首先将返回值存储在一个指定的位置,其次去执行finally块,最后再返回。在方法testFinally1中调用return前,先把result的值1存储在一个指定的位置,然后再去执行finally块中的代码,此时修改result的值将不会影响到程序的返回结果。testFinally2中,在调用return前先把s存储到一个指定的位置,由于s为引用类型,因此在finally中修改s将会修改程序的返回结果。


引申:出现在Java程序中的finally块是不是一定会被执行?

答案:不一定。

下面给出两个finally块不会被执行的例子:

1)、当程序进入try块之前就出现异常时,会直接结束,不会执行finally块中的代码,示例如下:

package com.js;
/**
 * 在try之前发生异常
 * @author jiangshuai
 */

public class Test{
	public static void testFinally1(){
		int result = 1/0;
		try {
			System.out.println("try block");
		} catch (Exception e) {
			System.out.println("catch block");
		}finally{
			System.out.println("finally block");
		}
	}
	public static void main(String[] args){
		testFinally1();
	}
}
运行结果:

Exception in thread "main" java.lang.ArithmeticException: / by zero
at com.js.Test.testFinally1(Test.java:9)
at com.js.Test.main(Test.java:19)
程序在执行1/0时会抛出异常,导致没有执行try块,因此finally块也就不会被执行。

2)、当程序在try块中强制退出时也不会去执行finally块中的代码,示例如下:

package com.js;
/**
 * 在try之前发生异常
 * @author jiangshuai
 */

public class Test{
	public static void testFinally1(){
		try {
			System.out.println("try block");
			System.exit(0);
		} catch (Exception e) {
			System.out.println("catch block");
		}finally{
			System.out.println("finally block");
		}
	}
	public static void main(String[] args){
		testFinally1();
	}
}
运行结果:

try block

上例在try块中通过调用System.exit(0)强制退出了程序,因此导致finally块中的代码没有被执行。


<think>我们正在讨论finally代码执行问题。用户询问finally是否一定会执行。 根据Java等语言的规定,finally在绝大多数情况下都会执行,但存在极少数特殊情况。 需要明确说明: 1. 正常情况下(包括try正常结束、catch捕获并处理异常)finally都会执行。 2. 在极端情况下finally可能执行,例如: - JVM非正常退出(如调用System.exit()) - 操作系统强制终止进程(如kill -9) - 守护线程在所有非守护线程结束时被终止,而finally可能来执行 - 死循环或线程被挂起导致无法进入finally 3. 另外,如果try或catch中抛出未被捕获的异常,finally还是会先执行再抛出。 因此,结论是:一般情况下finally都会执行,但在程序被强制终止等极端情况下执行。 同时,我们需要生成相关问题,这些问题应与finally相关,且基于我们的回答内容。 </think>### finally代码是否一定会执行? 在编程中(以Java为例),**finally代码在绝大多数情况下都会执行**[^1],但存在少数极端情况可能导致其执行。以下是详细分析: #### ✅ 正常情况下一定会执行 无论是否发生异常,finally都会执行: ```java try { // 可能出错的代码 int x = 10 / 2; // 正常执行 } catch (Exception e) { System.out.println("捕获异常"); } finally { System.out.println("finally执行"); // 必定输出 } ``` #### ✅ 发生异常时仍会执行 即使try或catch抛出异常: ```java try { int x = 10 / 0; // 抛出ArithmeticException } catch (Exception e) { throw new RuntimeException("新异常"); // 再次抛出 } finally { System.out.println("finally执行"); // 仍会输出 } ``` #### ️ 执行的极端情况(罕) 1. **JVM强制终止** 调用`System.exit(int)`直接终止JVM: ```java try { System.exit(1); // JVM立即终止 } finally { System.out.println("执行"); // 被跳过 } ``` 2. **操作系统级中断** 如Linux的`kill -9`强制杀死进程[^2]。 3. **底层硬件故障** 如电源中断、硬件崩溃等物理原因。 4. **守护线程未执行完** 当所有非守护线程结束时,JVM等守护线程的finally完成: ```java Thread daemon = new Thread(() -> { try { ... } finally { System.out.println("可能执行"); } // 守护线程 }); daemon.setDaemon(true); daemon.start(); ``` 5. **无限循环阻塞** finally前出现死循环: ```java try { while (true) {} // 无限循环 } finally { System.out.println("永执行"); } ``` #### 最终执行流程图 ```mermaid flowchart TD A[开始try-catch] --> B{执行try/catch} B -->|正常执行| C[执行finally] B -->|抛出异常| D[执行finally] C --> E[继续后续代码] D --> F[向上抛出异常] B -->|System.exit| G[JVM终止] --> H[跳过finally] B -->|硬件故障| I[进程崩溃] --> H ``` #### 最佳实践建议 1. **资源释放必用finally** 文件/网络连接等必须在finally中关闭[^3]: ```java FileInputStream file = null; try { file = new FileInputStream("test.txt"); } finally { if (file != null) file.close(); // 确保资源释放 } ``` 2. **避免finally中返回值** 返回值应放在try/catch中,否则会覆盖异常: ```java // 错误示例 try { return 1; } finally { return 2; } // 是返回2 ``` 3. **谨慎处理finally中的异常** finally中的异常会覆盖原始异常: ```java try { throw new IOException(); } finally { throw new RuntimeException(); } // 原始IOException丢失 ``` > 结:**finally在99%的场景下都会执行**,仅当程序被强制终止或硬件故障等极端情况才会跳过[^4]。它是资源清理的安全网,但需避免在其中放入关键业务逻辑。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值