线程终止的几种方式

前言

JDK 有一个 Deprecated 方法 stop,但是该方法存在一个问题,JDK 官方早已经不推 荐使用,其在后面的版本中有可能会被移除,根据官网的描述,该方法在关闭线程时可能不 会释放掉 monitor 的锁,所以强烈建议不要使用该方法结束线程。

正常关闭

A. 线程结束生命周期正常结束

线程运行结東,完成了自己的使命之后,就会正常退出,如果线程中的任务耗时比较短, 或者时间可控,那么放任它正常结束就好了。

B. 捕获中断信号关闭线程

我们通过 new Thread 的方式创建线程,这种方式看似很简单,其实它的派生成本是比 较高的,因此在一个线程中往往会循环地执行某个任务,比如心跳检査,不断地接收网络消 息报文等,系统决定退出的时候,可以借助中断线程的方式使其退出,示例代码如下:

public class InterruptThreadExit {
    public static void main(String[] args) {
        Thread t = new Thread(){
            @Override
            public void run() {
                System.out.println("I will start work.");
                while(!this.isInterrupted()){
                    //working.........
                }
                System.out.println("I will be exiting.");
            }
        };
        t.start();

        try {
            //五秒后结束线程
            TimeUnit.SECONDS.sleep(5);
            System.out.println("System will be shutdown.");
            t.interrupt();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

C. 使用 volatile 开关控制

由于线程的 interrupt 标识很有可能被擦除,或者逻辑单元中不会调用任何可中断方 法,所以使用 volatile 修饰的开关 flag 关闭线程也是一种常用的做法,具体如下:

public class FlagThreadExit {
    public static void main(String[] args)  {
        MyTast myTast = new MyTast();
        myTast.start();
        //5s后停止线程
        try {
            TimeUnit.SECONDS.sleep(5);
            System.out.println("System will be shutdown.");
            myTast.close();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

class MyTast extends Thread{
    private volatile boolean closed = false;
    @Override
    public void run() {
        System.out.println("I will start work.");
        while (!closed && !isInterrupted()) {
            // working.
        }
        System.out.println("I will be exiting.");
    }
    public void close(){
        this.closed = true;
        this.interrupt();
    }
}

异常退出

在一个线程的执行单元中,是不允许抛出 checked 异常的,不论 Thread 中的 run 方 法,还是 Runnable 中的 run 方法,如果线程在运行过程中需要捕获 checked 异常并且判 断是否还有运行下去的必要,那么此时可以将 checked 异常封装成 unchecked 异常 (RuntimeException)抛出进而结束线程的生命周期。

进程假死

相信很多程序员都会遇到进程假死的情况,所谓假死就是进程虽然存在,但没有日志输 出,程序不进行任何的作业,看起来就像死了一样,但事实上它是没有死的,程序之所以出 现这样的情况,绝大部分的原因就是某个线程阻塞了,或者线程出现了死锁的情况。使用jdk的jvisualvm 工具查看线程状态。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值