java try catch finally return执行
- public static int testBasic(){
- int i = 1;
- try{
- i++;
- System.out.println("try block, i = "+i);
- }catch(Exception e){
- i ++;
- System.out.println("catch block i = "+i);
- }finally{
- i = 10;
- System.out.println("finally block i = "+i);
- }
- return i;
- }
没错,会按照顺序执行,先执行try内代码段,没有异常的话进入finally,最后返回,那么输出如下:
try block, i = 2
finally block i = 10
main test i = 10
这个没有问题,如果我们把return语句放入try catch里又会怎么样呢
- public static int testBasic(){
- int i = 1;
- try{
- i++;
- System.out.println("try block, i = "+i);
- return i;
- }catch(Exception e){
- i ++;
- System.out.println("catch block i = "+i);
- return i;
- }finally{
- i = 10;
- System.out.println("finally block i = "+i);
- }
- }
输出结果是:
try block, i = 2
finally block i = 10
main test i = 2
代码顺序执行从try到finally,由于finally是无论如何都会执行的,所以try里的语句并不会直接返回。在try语句的return块中,return返回的引用变量并不是try语句外定义的引用变量i,而是系统重新定义了一个局部引用i’,这个引用指向了引用i对应的值,也就是2,即使在finally语句中把引用i指向了值10,因为return返回的引用已经不是i,而是i',所以引用i的值和try语句中的返回值无关了。
但是,这只是一部分,如果把i换成包装类型而不是基本类型呢,来看看输出结果怎样,示例如下:
- public static List<Object> testWrap(){
- List<Object> list = new ArrayList<>();
- try{
- list.add("try");
- System.out.println("try block");
- return list;
- }catch(Exception e){
- list.add("catch");
- System.out.println("catch block");
- return list;
- }finally{
- list.add("finally");
- System.out.println("finally block ");
- }
- }
打印结果如下:
try block
finally block
main test i = [try, finally]
可以看到,finally里对list集合的操作生效了,这是为什么呢。我们知道基本类型在栈中存储,而对于非基本类型是存储在堆中的,返回的是堆中的地址,因此内容被改变了。
好了,现在我们在finally里加一个return,看看语句是从哪里返回的。
- public static int testBasic(){
- int i = 1;
- try{
- i++;
- System.out.println("try block, i = "+i);
- return i;
- }catch(Exception e){
- i ++;
- System.out.println("catch block i = "+i);
- return i;
- }finally{
- i = 10;
- System.out.println("finally block i = "+i);
- return i;
- }
- }
输出结果如下:
try block, i = 2
finally block i = 10
main test i = 10
可以看到,是从finally语句块中返回的。可见,JVM是忽略了try中的return语句。但IDE中会对finally中加的return有黄色警告提示,这是为什么呢,在try里加入一行会执行异常的代码,如下:
- public static int testBasic(){
- int i = 1;
- try{
- i++;
- int m = i / 0 ;
- System.out.println("try block, i = "+i);
- return i;
- }catch(Exception e){
- i ++;
- System.out.println("catch block i = "+i);
- return i;
- }finally{
- i = 10;
- System.out.println("finally block i = "+i);
- return i;
- }
- }
打印结果如下:
catch block i = 3
finally block i = 10
main test i = 10
可以看到,因为finally中有return语句,try、catch中的异常被消化掉了,屏蔽了异常的发生,这与初期使用try、catch的初衷是相违背的,因此编译器也会提示警告。
那如果在finally中有异常发生,会对try、catch中的异常有什么影响呢?
- public static int testBasic(){
- int i = 1;
- try{
- i++;
- Integer.parseInt(null);
- System.out.println("try block, i = "+i);
- return i;
- }catch(Exception e){
- String.valueOf(null);
- System.out.println("catch block i = "+i);
- return i;
- }finally{
- i = 10;
- int m = i / 0;
- System.out.println("finally block i = "+i);
- }
- }
这里我们在try、catch里强行加上异常语句,打印结果如下:
Exception in thread "main" java.lang.ArithmeticException: / by zero
at tryandcatch.TryAndCatch.testBasic(TryAndCatch.java:25)
at tryandcatch.TryAndCatch.main(TryAndCatch.java:45)
这个提示表示的是finally里的异常信息,也就是说一旦finally里发生异常,try、catch里的异常信息即被消化掉了,也达不到异常信息处理的目的。
总结以上测试:
1、finally语句总会执行
2、如果try、catch中有return语句,finally中没有return,那么在finally中修改除包装类型和静态变量、全局变量以外的数据都不会对try、catch中返回的变量有任何的影响(包装类型、静态变量会改变、全局变量)
3、尽量不要在finally中使用return语句,如果使用的话,会忽略try、catch中的返回语句,也会忽略try、catch中的异常,屏蔽了错误的发生
4、finally中避免再次抛出异常,一旦finally中发生异常,代码执行将会抛出finally中的异常信息,try、catch中的异常将被忽略
所以在实际项目中,finally常常是用来关闭流或者数据库资源的,并不额外做其他操作。
java try catch finally return执行的更多相关文章
- 异常 try catch finally return 执行关系 MD
Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...
- Java基础知识强化之IO流笔记06:有return的情况下try catch finally的执行顺序
1. 给出结论: (1)不管有木有出现异常,finally块中代码都会执行:(2)当try和catch中有return时,finally仍然会执行:(3)finally是在return后面的表达式运算 ...
- Java中的try,catch(Exception e),finally及return执行顺序
结论: ①就算之前return,finally也会执行 ②finally的计算结果不影响之前的return值 ③finally的return值一定是最后的返回结果,因此将return放入finally ...
- 【Java疑难杂症】有return的情况下try catch finally的执行顺序
有这样一个问题,异常处理大家应该都不陌生,类似如下代码: public class Test { public static void main(String[] args) { int d1 = 0 ...
- Java_try,catch,finally return之间的执行顺序
以往认为函数只要执行到return语句便会返回结果并终止,然而这时错误的,因为这存在特例. 掌握下面几条原则就可以完全解决“当try.catch.finally遭遇return”的问题. 原则:1.f ...
- JVM关键字try、catch、finally、return执行过程
关键字:jvm try catch finally return.指令 finally相当于在所有方法返回之前执行一次 finally中含有return其中finally中return会覆盖try和c ...
- 有return的情况下try catch finally的执行顺序(转)
结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...
- try catch finally的执行顺序(有return的情况下)
结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...
- 有return的情况下try catch finally的执行顺序
结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...
随机推荐
- ThinkJava-压缩
尽管存在许多种压缩算恙,但是Zip和GZIP可能是最常用的.因此我们可以很容易地使用多 种可读写这些格式的工具来操纵我们的压缩数据. 1 用GZIP进行简单压缩 GZIP接口非常简单, 因此如果我 ...
- Java 从原字符串中截取一个新的字符串 subString()
Java 手册 substring public String substring(int beginIndex) 返回一个新的字符串,它是此字符串的一个子字符串.该子字符串从指定索引处的字符开始,直 ...
- Java之MD5加密
一.Message Digest Algorithm MD5(中文名为消息摘要算法第五版)为计算机安全领域广泛使用的一种散列函数,用以提供消息的完整性保护.该算法的文件号为RFC 1321(R.Riv ...
- Javascript继承机制的设计思想
转自:http://www.ruanyifeng.com/blog/2011/06/designing_ideas_of_inheritance_mechanism_in_javascript.htm ...
- 正规式->最小化DFA说明
整体的步骤是三步: 一,先把正规式转换为NFA(非确定有穷自动机), 二,在把NFA通过"子集构造法"转化为DFA, 三,在把DFA通过"分割法"进行最小化 ...
- Api2Doc生成 Restful API 文档
1,引入maven <dependency> <groupId>com.github.terran4j</groupId> <artifactId>te ...
- css常用属性总结:背景background上篇
1.在前面一节中我们提到了color的使用,类似于前景色,我们同样可以为元素的背景声明颜色,可以使用background-color属性, 它接受任何有效的颜色值.先看看它的语法. backgroun ...
- android在开发过程中的数据库存储位置
1
- linux shell脚本编程笔记(五): 重定向
I/O重定向 简述: 默认情况下始终有3个"文件"处于打开状态, stdin (键盘), stdout (屏幕), and stderr (错误消息输出到屏幕上). 这3个文件和其 ...
- 文件上传:input file FileReader
js: window.onload = function () { var input = document.getElementById('input-file'), info = document ...