前言

这看似一个完全没有意义的问题,但是如果你是从搜索引擎过来的话,那么说明你碰到过这个问题。
线程执行完不就退出了,说停止有什么意义?
当然有意义,意义在于,一般创建线程后,如果是一次性的线程,执行结束就可以了,不用管它。
如果是一个一直需要保持运行,而需要在某一时刻才需要停止的线程,就需要关注线程是如何退出的。

退出方式

  1. 退出标志: 使用退出标志,使线程正常退出,也就是当run方法完成后线程终止。
  2. interrupt: 使用interrupt方法中断线程。
  3. 可以但不推荐: 不推荐使用 stop、suspend及resume 方法。stop 相当于电脑断电关机一样,是不安全的方法。

退出标志

方式:使用一个标志不控制线程是否需要继续执行。
常驻的业务线程当中一般都会写循环,如果不写循环,一句话能搞定的事或者需要调用才执行的事,就没必要再开线程来处理。
stop方法已经过时,不推荐使用。

开启多线程时,运行代码通常是循环结构,只要控制住循环,就可以让run方法结束,也就是线程结束。
原理:只要循环终止了,线程也就终止了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class StopThread implements Runnable{

private volatile boolean flag = true;

@Override
public void run() {
// flag 来控制线程的状态
while (flag) {
System.out.println(Thread.currentThread().getName() + "...run");
}
System.out.println("...stop");
}

public void set() {
flag = false;
}
}

使用了一个关键字 volatile,保证当前 flag 在多核CPU 下的可见性。

interrupt 方式

方式:主动中断线程。

上面不是说不要使有stop这种方式来停止吗,这里还用 interrupt 来主动中断。
interrupt 和 stop 不同,使用这个方法并不会让线程立即中断,而是给线程打一个标志,线程会在合适的时机退出。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
package com.liukai.thread.stop;

public class MyThread extends Thread {
public void run(){
try {
for(int i=0; i<5000; i++){
if (i == 100) {
// 打印这一句话后,后续还会打印 i= 后面的数字,证明线程并没有立即停止
System.out.println("主动中断线程");
Thread.currentThread().interrupt();
}
System.out.println("i="+(i+1));
Thread.sleep(100);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

总结

以上两种方式,就是退出线程安全的方式,可以自行验证这两个方法。