JDK有一个Deprecated方法stop,但是该方法存在一个问题,JDK官方早已经不推荐使用,其在后面的版本中有可能会被移除,根据官网的描述,该方法在关闭线程时可能不会释放掉monitor的锁,所以强烈建议不要使用该方法结束线程。
1 正常关闭
-
线程结束生命周期正常结束。
线程运行结束,完成了自己的使命之后,就会正常退出,如果线程中的任务耗时比较短,或者时间可控,那么放任它正常结束就好了。 -
捕获中断信号关闭线程,比如下面这段代码:
上面的代码是通过检查线程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();
}
- .使用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等工具