Thread类API详解(四)-如何关闭一个线程

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

1 正常关闭

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

  2. 捕获中断信号关闭线程,比如下面这段代码:
    上面的代码是通过检查线程interrupt的标识来决定是否退出的,如果在线程中执行某个可中断方法,则可以通过捕获中断信号来决定是否退出。

public static void main(String[] args) throws InterruptedException {
    Thread thread = new Thread(() -> {
        while (!Thread.currentThread().isInterrupted()) {
            System.out.println("MyThead is working");
        }
        System.out.println("MyThead will exist");
    }, "MyThead");

    thread.start();
    // 主线程1s 后打断线程
    TimeUnit.SECONDS.sleep(1);
    thread.interrupt();
}

又或者,在线程中存在可中断的方法,当外部调用该线程的interrupt,这些中断方法也会捕获到异常,通过这个方式关闭:

public static void main(String[] args) throws InterruptedException {
    Thread thread = new Thread(() -> {
        while (true){
            System.out.println("MyThead is working");
            try {
                TimeUnit.MILLISECONDS.sleep(500);
            } catch (InterruptedException e) {
                // 捕获到打断标识
                System.out.println("MyThead will exist");
                break;
            }
        }
    }, "MyThead");

    thread.start();
    // 主线程1s 后打断线程
    TimeUnit.SECONDS.sleep(1);
    thread.interrupt();
}
  1. .使用volatile开关控制
    由于线程的interrupt标识很有可能被擦除,或者逻辑单元中不会调用任何可中断方法,所以使用volatile修饰的开关flag关闭线程也是一种常用的做法,参考代码如下:
public class Demo3 {
  static class MyThead extends Thread{
      /**
       * 是否关闭线程的标记,默认为false
       */
      private volatile boolean closed = false;

      @Override
      public void run() {
          System.out.println("I will start work");
          while (!closed && !isInterrupted()){
              System.out.println("i am working.");
          }
          System.out.println("I will be exiting.");
      }
      public void close()
      {
          this.closed = true;
          this.interrupt();
      }
  }

  public static void main(String[] args) throws InterruptedException {
      MyThead t = new MyThead();
      t.start();
      TimeUnit.MILLISECONDS.sleep(200);
      t.close();
  }
}

2 异常退出

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

3 进程假死

所谓假死就是进程虽然存在,但没有日志输出,程序不进行任何的作业,看起来就像死了一样,但事实上它是没有死的,程序之所以出现这样的情况,绝大部分的原因就是某个线程阻塞了,或者线程出现了死锁的情况。
我们需要借助一些工具来帮助诊断,比如jstack、jconsole、jvisualvm等工具

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值