finally块不被执行的情况总结

finally块的作用

通常用于处理善后工作。当try块里出现异常时,会立即跳出try块,到catch块匹配对应的异常,执行catch块里的语句。此时,可能在try块里存在打开的文件没关闭,连接的网络没断开,这部分资源是GC所不能自动处理的,所以finally的作用就是将它们及时释放回收。

finally块不被执行的情况,总共有3种:不进入try块、程序中止、线程中止(带finally块的是守护线程,其非守护线程都执行完毕)。

1. 未执行try块

对于try-catch-finally或者try-finally,如果不能进入try块,则finally块也无法进入。

public class Test {
    public static void main(String[] args) {
        boolean flag = false;
        if(flag) {
          try {
              System.out.println("enter try block");
           } finally {
              System.out.println("enter finally block");
           }
        }
   }
}
 /******************
 控制台无输出
 *******************/

2. System.exit()

public class Test {
    public static void main(String[] args) {
       try {
            System.out.println("enter try block");
            System.exit();
       } finally {
            System.out.println("enter finally block");
       }
    }
}
/*****************
控制台打印如下
enter try block
******************/

3. try块陷入无限循环

public class Test {
    public static void main(String[] args) {
       try {
            while(true){
                System.out.println("enter try block");
            }
       } finally {
            System.out.println("enter finally block");
       }
    }
}
/*****************
 完蛋
 *****************/

4. 守护(daemon)线程被中止时

Java线程分为两类,守护线程和非守护线程。当所有的非守护线程都终止时,无论守护线程存不存在,虚拟机都会kill掉所有的守护线程从而中止程序。

虚拟机中,执行main方法的线程就是一个非守护线程,垃圾回收则是另一个守护线程,main执行完,则程序中止,而不管垃圾回收线程是否中止。

所以,如果守护线程中存在finally代码块,那么当所有的非守护线程中止时,守护线程被kill掉,其finally代码块是不会执行的。

public class Test {
    public static void main(String[] args) {
        //main是一个非守护线程
        Thread thread = new Thread(new Task());
        thread.setDaemon(true); //设置thread为守护线程
        thread.start();
        TimeUnit.SECONDS.sleep(5); //阻塞5s.
    }
}
class Task implements Runnable {
    @Override
    public void run() {
         System.out.println("enter run()");
        try {
            System.out.println("enter try block");
        } catch(InterruptedException e) {
            System.out.println("enter catch block");
        } finally {
            System.out.println("enter finally block");
        }
    }
}
/*******************
控制台打印如下
enter run()
enter try block
enter try finally block
********************/

上述代码,语句 TimeUnit.SECONDS.sleep(5); 会使main线程阻塞5秒,足够线程thread执行。

如果将该语句注释,非守护线程main线程执行完 thread.start(); 这行后,存在三种情况:①CPU时间片还是交给main线程,则非守护线程执行完毕,守护线程thread就会被终止,finally块不执行;②CPU时间片交给thread线程,但是thread线程刚执行完try块,就得交付时间片给main,main已经无语句执行,就会结束,导致守护线程thread也要结束,finally块不执行;③CPU时间片交付thread线程,thread线程完全执行,finally块被执行。

其他迷惑性选项

(1)当try块里面包含有break,该次try块结束后,finally块也会执行。

public class Test {
    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            try {
                if (i == 2) {
                    break;
                }
            } finally {
                System.out.print(i);
            }
        }
    }
}

/*************
 输出结果:012
**************/

(2)当try块里面包含有return,该次try块结束后,finally块也会执行。

public class Test {
    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            try {
                if (i == 2) {
                    return;
                }
            } finally {
                System.out.print(i);
            }
        }
    }
}

/*************
 输出结果:012
**************/

(3)当try块里面包含有continue,该次try块结束后,finally块也会执行。

public class Test {
    public static void main(String[] args) {
        for (int i = 0; i < 5; i++) {
            try {
                if (i == 2) {
                    continue;
                }
            } finally {
                System.out.print(i);
            }
        }
    }
}

/*************
 输出结果:01234
**************/
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: try catch finally执行顺序是先执行 try 中的代码,如果出现异常则跳转到对应的 catch 执行 catch 中的代码,最后无论是否出现异常,都会执行 finally 中的代码。 ### 回答2: try catch finally 执行顺序是先执行 try 语句中的代码,如果没有发生异常则跳过 catch 语句,直接执行 finally 语句中的代码。如果在 try 语句中发生了异常,程序会跳出 try ,然后寻找匹配的 catch 来处理异常。如果找到了匹配的 catch ,则执行这个中的代码,并跳过 finally 。如果没有找到匹配的 catch ,则异常会传递到上层调用者,直到找到了匹配的 catch 或程序结束。 无论是否发生异常,finally 语句中的代码都会被执行,即使在 try 或 catch 中有 return 语句。当异常被抛出时,会先执行 finally 的代码,然后再将异常传递给上层调用者。 总结起来,try 语句中的代码首先被执行,如果没有发生异常,则执行 finally 语句中的代码;如果发生异常,则先执行 finally 语句中的代码,然后执行匹配的 catch 语句中的代码,最后再执行 finally 语句中的代码。无论是否发生异常,finally 语句中的代码都会被执行。 这种执行顺序的设计可以确保在程序执行过程中资源的释放和清理操作,即使发生了异常也能够得到处理。 ### 回答3: try catch finally 是一种错误处理机制。在程序执行过程中,当发生错误时,try语句中的代码会被执行,如果没有错误发生,catch和finally语句就不会执行。 try语句中的代码被执行时,如果出现异常,程序会立即转到与之对应的catch语句。catch语句负责处理异常,并提供对应的异常处理逻辑。在catch语句中,可以根据捕获到的异常类型进行相应的处理,例如输出错误信息、记录日志、回滚事务等。 无论是否发生异常,finally语句中的代码都会被执行。这意味着,无论try中的代码是否成功执行finally语句里的代码都会被执行到。finally语句通常用于释放资源、回收内存等操作,在代码中确保必须执行的一段逻辑。 try catch finally执行顺序如下: 1. 程序首先执行try语句中的代码。 2. 如果try语句中的代码没有发生异常,catch语句将被跳过。 3. 如果try语句中的代码发生异常,运行时系统会查找与之对应的catch语句,并执行相应的异常处理逻辑。 4. catch语句执行完毕后,程序会继续执行finally语句中的代码。 5. finally语句中的代码执行完毕后,程序继续执行try catch finally之后的代码。 总的来说,try catch finally执行顺序是:先执行try语句,再执行catch语句(如果有异常发生),最后执行finally语句

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值