一.resume()、suspend()、stop()
这三个方法分别是重启线程、暂停线程、终止线程。
这三个方法已经弃用。
弃用原因:
1.suspend(),在调用该方法暂停线程的时候,线程由running状态变成blocked,需要等待resume方法将其重新变成runnable。
而线程由running状态变成blocked时,只释放了CPU资源,没有释放锁资源,可能出现死锁。
比如:线程A拿着锁1被suspend了进入了blocked状态,等待线程B调用resume将线程A重新runnable。但是线程B一直在lock pool中等待锁1,线程B要拿到锁1才能running去执行resumeA。这就死锁了。
2.stop(),调用stop方法无论run()中的逻辑是否执行完,都会释放CPU资源,释放锁资源。这会导致线程不安全。
比如:线程A的逻辑是转账(获得锁,1号账户减少100元,2号账户增加100元,释放锁),那线程A刚执行到1号账户减少100元就被调用了stop方法,释放了锁资源,释放了CPU资源。1号账户平白无故少了100元。一场撕逼大战开始了。
二、相对靠谱的暂停、重启、停止线程的方案
用flag来实现
记得flag字段要加volatile
暂停、挂起的逻辑实现只要是用释放CPU、释放锁的方式来实现就行了,不一定非要用wait()、notify(),比如还能用ReentrantLock、Condition、lock()、signalAll()
public class StopThreadTest {
public static void main(String[] args) {
StopThread r1 = new StopThread();
Thread t1 = new Thread(r1,"My Thread");
t1.start();
try {
Thread.sleep(1000);
r1.mysuspend();
System.out.println("Suspending thread.");
Thread.sleep(1000);
r1.myresume();
System.out.println("Resuming thread.");
Thread.sleep(1000);
r1.mysuspend();
System.out.println("Suspending thread.");
Thread.sleep(1000);
r1.myresume();
System.out.println("Resuming thread.");
Thread.sleep(1000);
r1.mysuspend();
System.out.println("Suspending thread.");
r1.mystop();
}catch (Exception e){
e.printStackTrace();
}
try{
t1.join();
}catch (Exception e){
System.out.println("Main thread Interrupted");
}
System.out.println("Main thread exiting");
}
}
class StopThread implements Runnable{
volatile boolean suspended;
volatile boolean stopped;
StopThread(){
suspended = false;
stopped = false;
}
public void run(){
System.out.println(Thread.currentThread().getName());
try {
for(int i=1; i <1000;i++){
System.out.println(i+" ");
if((i%10)==0){
System.out.println();
Thread.sleep(250);
}
synchronized (this){
while (suspended){
wait();
}
if(stopped)
break;
}
}
}catch (Exception e){
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+" exiting.");
}
synchronized void mystop(){
stopped = true;
suspended = false;
notify();
}
synchronized void mysuspend(){
suspended = true;
}
synchronized void myresume(){
suspended = false;
notify();
}
}
My Thread
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
Suspending thread.
Resuming thread.
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
Suspending thread.
Resuming thread.
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
Suspending thread.
My Thread exiting.
Main thread exiting
Process finished with exit code 0
三、interrupt
1.Thread类的静态方法interrupted
该方法只是把当前线程设置一个被打断的标记。换句话说,这个方法只能是线程A自己打断自己。
2.thread对象的interrupt方法
该方法,只要拿到一个thread对象,就可以调用该thread对象的Interrupt方法,标记这个thread线程为被打断。
换句话说,A线程内,只要A线程能拿到B线程的thread对象,A线程可以去调用B线程的Interrupt()打断B线程。
而interrupted()则是只能B现在自己把自己打断。这是interrupted()和Interrupt()最明显的区别。
3.打断机制
interrupt()或interrupted()的作用是通知线程应该中断了,到底中断还是继续运行,应该由被通知的线程自己处理。当对一个线程,调用 interrupt() 时,
1.如果线程处于被阻塞状态(例如处于sleep, wait, join 等状态),那么线程将立即退出被阻塞状态,并抛出一个InterruptedException异常。
2.synchronized在等锁的过程中是不能被interrupt的,等于说如果产生了死锁,则不可能被interrupt中断抛InterruptedException异常。它只是被设置了中断标记,得获取到锁过后再处理中断。ReentrantLock的lock.lock()也一样。
但是ReentrantLock的lock.lockInterruptibly()在等锁的过程中可以立即响应interrupt,抛出InterruptedException异常,如上方3.1所述
3.如果线程处于正常活动状态,那么会将该线程的中断标志设置为 true,仅此而已。被设置中断标志的线程将继续正常运行,不受影响。如果线程A需要被interrupt()方法结束,需要线程A自己一直查看自己的终端标志(thread.isInterrupted())是否为true,自己写逻辑为true时做线程结束的逻辑。
思考:
上述1有个是否释放锁资源的问题,我理解的是:
如果线程A用是synchronized获取了锁,然后sleep了,在sleep的时候被interrupt了,本来sleep就没释放锁资源,这个时候interrupt这个A线程,在catinterruptedExcption异常中应该也是还具备锁资源的。
测试代码:
上面这个测试代码只是解释,如果catch (InterruptedException e)在synchronized里面,那这个catch里面还是拿着锁资源的。
实际代码中,鑫大爷特别强调:不要哈戳戳的在synchronized内捕获InterruptedException异常还不退出来,还不往外抛,还继续做以后后面的逻辑,不要这样搞,不要这样哈戳戳的。
正确的操作应该是如下:在synchronized外面捕获InterruptedException异常,当sleep被打断后,不会再执行sleep后面的代码,而是结束synchronized代码块,释放锁,然后在走catch (InterruptedException e)的逻辑