java中finally语句快会不会执行的问题

一、先看下面的代码

public static void main(String[] args) {

        try {
            System.out.println("how's the weather today?");
        } catch (Exception e) {
            System.out.println("i don't know");
        }finally{
            System.out.println("weather is fine");
        }

运行结果:

 how's the weather today?
    weather is fine

 public static void main(String[] args) {

        try {
            System.out.println("how's the weather today?");
            return;
        } catch (Exception e) {
            System.out.println("i don't know");
            return;
        }finally{
            System.out.println("weather is fine");
        }
运行结果:

 how's the weather today?
    weather is fine

public static void main(String[] args) {

        try {
            System.out.println("how's the weather today?");
           <span style="color:#ff0000;"> System.exit(0);</span>
        } catch (Exception e) {
            System.out.println("i don't know");
        }finally{
            System.out.println("weather is fine");
        }

运行结果:

  how's the weather today?

 观看输出发现finally块并没有被执行,JVM都退出了,还怎么运行呢。所以以后有人问的话,可以这样说,在JVM正常运行的情况下,finally块一定会执行。

二、在下面情况finally会被忽略

这种情况就是在并发程序中,将finally语句块所在的线程(假设为threadFin)设置成后台进程,即threadFin.setDaemon(true);此外,调用后台线程的方法应该是非后台线程的形式,这样,当非后台线程终止结束时,这个方法所调用的后台线程就会被JVM立即关闭,后台线程中的方法将不会去执行finally语句。

package com.du.concurrent;  
  
import java.util.concurrent.TimeUnit;  
  
public class ADaemon implements Runnable {  
  
    @Override  
    public void run() {  
        // TODO Auto-generated method stub  
        try{  
            System.out.print("Starting ADaemon");  
            TimeUnit.MILLISECONDS.sleep(1);  
        }catch(InterruptedException e){  
            System.out.print("Exiting via InterruptedException");  
        }finally{  
            System.out.print("the finally really run?");  
        }  
    }  
  
    public static void main(String[] args) {  
        // TODO Auto-generated method stub  
        ADaemon ad = new ADaemon();  
        Thread thread = new Thread(ad);  
        <span style="color:#cc0000;">thread.setDaemon(true); //如果注释掉这行,finally语句块将会执行  </span>
        thread.start();  
    }  
}  
显然,后台进程的关闭是因为main()函数的退出,即非后台线程的结束而被强制执行的,所以,这种方式对后台进程的管理是极其不友好的。
三、thread.setDaemon(true);作用

定义:守护线程--也称“服务线程”,在没有用户线程可服务时会自动离开。优先级:守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。
设置:通过setDaemon(true)来设置线程为“守护线程”;将一个用户线程设置为守护线程的方式是在 线程对象创建 之前 用线程对象的setDaemon方法。
example: 垃圾回收线程就是一个经典的守护线程,当我们的程序中不再有任何运行的Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于实时监控和管理系统中的可回收资源。
生命周期:守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。也就是说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。那Java的守护线程是什么样子的呢。当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个或以上的非守护线程则JVM不会退出。



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值