线程状态图
说明: 线程共包括以下5种状态。
新建状态(New) : 线程对象被创建后,就进入了新建状态。例如,Thread thread = new Thread()。
就绪状态(Runnable): 也被称为“可执行状态”。线程对象被创建后,其它线程调用了该对象的start()方法,从而来启动该线程。例如,thread.start()。处于就绪状态的线程,随时可能被CPU调度执行。
运行状态(Running) : 线程获取CPU权限进行执行。需要注意的是,线程只能从就绪状态进入到运行状态。
阻塞状态(Blocked) : 阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。阻塞的情况分三种: (01) 等待阻塞 – 通过调用线程的wait()方法,让线程等待某工作的完成。 (02) 同步阻塞 – 线程在获取synchronized同步锁失败(因为锁被其它线程所占用),它会进入同步阻塞状态。 (03) 其他阻塞 – 通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。
死亡状态(Dead) : 线程执行完了或者因异常退出了run()方法,该线程结束生命周期。
这5种状态涉及到的内容包括Object类, Thread和synchronized关键字。 Object类,定义了wait(), notify(), notifyAll()等休眠/唤醒函数。 Thread类,定义了一些列的线程操作函数。例如,sleep()休眠函数, interrupt()中断函数, getName()获取线程名称等。 synchronized,是关键字;它区分为synchronized代码块和synchronized方法。synchronized的作用是让线程获取对象的同步锁。
Thread和Runnable简介 Runnable 是一个接口,该接口中只包含了一个run()方法。它的定义如下:
public interface Runnable { public abstract void run(); } Runnable的作用,实现多线程。我们可以定义一个类A实现Runnable接口;然后,通过new Thread(new A())等方式新建线程。
Thread 是一个类。Thread本身就实现了Runnable接口。它的声明如下:
public class Thread implements Runnable {} Thread的作用,实现多线程。
Thread和Runnable的异同点 Thread 和 Runnable 的相同点:都是“多线程的实现方式”。 Thread 和 Runnable 的不同点: Thread 是类,而Runnable是接口;Thread本身是实现了Runnable接口的类。我们知道“一个类只能有一个父类,但是却能实现多个接口”,因此Runnable具有更好的扩展性。 此外,Runnable还可以用于“资源的共享”。即,多个线程都是基于某一个Runnable对象建立的,它们会共享Runnable对象上的资源。 通常,建议通过“Runnable”实现多线程!
Thread和Runnable的多线程示例
Thread的多线程示例 下面通过示例更好的理解Thread和Runnable,借鉴网上一个例子比较具有说服性的例子。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class MyThread extends Thread {
private int ticket=10 ;
public void run () {
for (int i=0 ;i<20 ;i++){
if (this .ticket>0 ){
System.out.println(this .getName()+" 卖票:ticket" +this .ticket--);
}
}
}
};
public class ThreadTest {
public static void main (String[] args) {
MyThread t1=new MyThread();
MyThread t2=new MyThread();
MyThread t3=new MyThread();
t1.start();
t2.start();
t3.start();
}
}
运行结果:
Thread-0 卖票:ticket10 Thread-1 卖票:ticket10 Thread-2 卖票:ticket10 Thread-1 卖票:ticket9 Thread-0 卖票:ticket9 Thread-1 卖票:ticket8 Thread-2 卖票:ticket9 Thread-1 卖票:ticket7 Thread-0 卖票:ticket8 Thread-1 卖票:ticket6 Thread-2 卖票:ticket8 Thread-1 卖票:ticket5 Thread-0 卖票:ticket7 Thread-1 卖票:ticket4 Thread-2 卖票:ticket7 Thread-1 卖票:ticket3 Thread-0 卖票:ticket6 Thread-1 卖票:ticket2 Thread-2 卖票:ticket6 Thread-2 卖票:ticket5 Thread-2 卖票:ticket4 Thread-1 卖票:ticket1 Thread-0 卖票:ticket5 Thread-2 卖票:ticket3 Thread-0 卖票:ticket4 Thread-2 卖票:ticket2 Thread-0 卖票:ticket3 Thread-2 卖票:ticket1 Thread-0 卖票:ticket2 Thread-0 卖票:ticket1
结果说明: (01) MyThread继承于Thread,它是自定义个线程。每个MyThread都会卖出10张票。 (02) 主线程main创建并启动3个MyThread子线程。每个子线程都各自卖出了10张票。
Runnable的多线程示例 下面,我们对上面的程序进行修改。通过Runnable实现一个接口,从而实现多线程。
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
class MyThread implements Runnable {
private int ticket=10 ;
public void run () {
for (int i=0 ;i<20 ;i++){
if (this .ticket>0 ){
System.out.println(Thread.currentThread().getName()+" 卖票:ticket" +this .ticket--);
}
}
}
};
public class RunnableTest {
public static void main (String[] args) {
MyThread mt=new MyThread();
Thread t1=new Thread(mt);
Thread t2=new Thread(mt);
Thread t3=new Thread(mt);
t1.start();
t2.start();
t3.start();
}
}
运行结果:
Thread-0 卖票:ticket10 Thread-2 卖票:ticket8 Thread-1 卖票:ticket9 Thread-2 卖票:ticket6 Thread-0 卖票:ticket7 Thread-2 卖票:ticket4 Thread-1 卖票:ticket5 Thread-2 卖票:ticket2 Thread-0 卖票:ticket3 Thread-1 卖票:ticket1
结果说明: (01) 和上面“MyThread继承于Thread”不同;这里的MyThread实现了Thread接口。 (02) 主线程main创建并启动3个子线程,而且这3个子线程都是基于“mt这个Runnable对象”而创建的。运行结果是这3个子线程一共卖出了10张票。这说明它们是共享了MyThread接口的。
start() 和 run()的区别说明 start() : 它的作用是启动一个新线程,新线程会执行相应的run()方法。start()不能被重复调用。 run() : run()就和普通的成员方法一样,可以被重复调用。单独调用run()的话,会在当前线程中执行run(),而并不会启动新线程!
下面以代码来进行说明。1
2
3
4
5
6
class MyThread extends Thread {
public void run () {
...
}
};
MyThread mythread = new MyThread();
mythread.start()会启动一个新线程,并在新线程中运行run()方法。 而mythread.run()则会直接在当前线程中运行run()方法,并不会启动一个新线程来运行run()。
start() 和 run()的区别示例
下面,通过一个简单示例演示它们之间的区别。源码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class MyThread extends Thread {
public MyThread (String name) {
super (name);
}
public void run () {
System.out.println(Thread.currentThread().getName()+" is running" );
}
};
public class Demo {
public static void main (String[] args) {
Thread mythread=new MyThread("mythread" );
System.out.println(Thread.currentThread().getName()+" call mythread.run()" );
mythread.run();
System.out.println(Thread.currentThread().getName()+" call mythread.start()" );
mythread.start();
}
}
运行结果:
main call mythread.run() main is running main call mythread.start() mythread is running
结果说明: (01) Thread.currentThread().getName()是用于获取“当前线程”的名字。当前线程是指正在cpu中调度执行的线程。 (02) mythread.run()是在“主线程main”中调用的,该run()方法直接运行在“主线程main”上。 (03) mythread.start()会启动“线程mythread”,“线程mythread”启动之后,会调用run()方法;此时的run()方法是运行在“线程mythread”上。
synchronized原理
在java中,每一个对象有且仅有一个同步锁。这也意味着,同步锁是依赖于对象而存在。 当我们调用某对象的synchronized方法时,就获取了该对象的同步锁。例如,synchronized(obj)就获取了“obj这个对象”的同步锁。 不同线程对同步锁的访问是互斥的。也就是说,某时间点,对象的同步锁只能被一个线程获取到!通过同步锁,我们就能在多线程中,实现对“对象/方法”的互斥访问。 例如,现在有两个线程A和线程B,它们都会访问“对象obj的同步锁”。假设,在某一时刻,线程A获取到“obj的同步锁”并在执行一些操作;而此时,线程B也企图获取“obj的同步锁” —— 线程B会获取失败,它必须等待,直到线程A释放了“该对象的同步锁”之后线程B才能获取到“obj的同步锁”从而才可以运行。
synchronized基本规则
我们将synchronized的基本规则总结为下面3条,并通过实例对它们进行说明。 第一条: 当一个线程访问“某对象”的“synchronized方法”或者“synchronized代码块”时,其他线程对“该对象”的该“synchronized方法”或者“synchronized代码块”的访问将被阻塞。 第二条: 当一个线程访问“某对象”的“synchronized方法”或者“synchronized代码块”时,其他线程仍然可以访问“该对象”的非同步代码块。 第三条: 当一个线程访问“某对象”的“synchronized方法”或者“synchronized代码块”时,其他线程对“该对象”的其他的“synchronized方法”或者“synchronized代码块”的访问将被阻塞。
第一条 当一个线程访问“某对象”的“synchronized方法”或者“synchronized代码块”时,其他线程对“该对象”的该“synchronized方法”或者“synchronized代码块”的访问将被阻塞。 下面是“synchronized代码块”对应的演示程序。
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
class MyRunable implements Runnable {
@Override
public void run () {
synchronized (this ) {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName() + " loop " + i);
}
} catch (InterruptedException ie) {
}
}
}
}
public class Demo1_1 {
public static void main (String[] args) {
Runnable demo = new MyRunable();
Thread t1 = new Thread(demo, "t1" );
Thread t2 = new Thread(demo, "t2" );
t1.start();
t2.start();
}
}
运行结果:
t1 loop 0 t1 loop 1 t1 loop 2 t1 loop 3 t1 loop 4 t2 loop 0 t2 loop 1 t2 loop 2 t2 loop 3 t2 loop 4
结果说明: run()方法中存在“synchronized(this)代码块”,而且t1和t2都是基于”demo这个Runnable对象”创建的线程。这就意味着,我们可以将synchronized(this)中的this看作是“demo这个Runnable对象”;因此,线程t1和t2共享“demo对象的同步锁”。所以,当一个线程运行的时候,另外一个线程必须等待“运行线程”释放“demo的同步锁”之后才能运行。
如果你确认,你搞清楚这个问题了。那我们将上面的代码进行修改,然后再运行看看结果怎么样,看看你是否会迷糊。修改后的源码如下:
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
class MyThread extends Thread {
public MyThread (String name) {
super (name);
}
@Override
public void run () {
synchronized (this ) {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName() + " loop " + i);
}
} catch (InterruptedException ie) {
}
}
}
}
public class Demo1_2 {
public static void main (String[] args) {
Thread t1 = new MyThread("t1" );
Thread t2 = new MyThread("t2" );
t1.start();
t2.start();
}
}
代码说明: 比较Demo1_2 和 Demo1_1,我们发现,Demo1_2中的MyThread类是直接继承于Thread,而且t1和t2都是MyThread子线程。 幸运的是,在“Demo1_2的run()方法”也调用了synchronized(this),正如“Demo1_1的run()方法”也调用了synchronized(this)一样! 那么,Demo1_2的执行流程是不是和Demo1_1一样呢? 运行结果:
t1 loop 0 t2 loop 0 t1 loop 1 t2 loop 1 t1 loop 2 t2 loop 2 t1 loop 3 t2 loop 3 t1 loop 4 t2 loop 4
结果说明: 如果这个结果一点也不令你感到惊讶,那么我相信你对synchronized和this的认识已经比较深刻了。否则的话,请继续阅读这里的分析。 synchronized(this)中的this是指“当前的类对象”,即synchronized(this)所在的类对应的当前对象。它的作用是获取“当前对象的同步锁”。 对于Demo1_2中,synchronized(this)中的this代表的是MyThread对象,而t1和t2是两个不同的MyThread对象,因此t1和t2在执行synchronized(this)时,获取的是不同对象的同步锁。对于Demo1_1对而言,synchronized(this)中的this代表的是MyRunable对象;t1和t2共同一个MyRunable对象,因此,一个线程获取了对象的同步锁,会造成另外一个线程等待。
第二条 当一个线程访问“某对象”的“synchronized方法”或者“synchronized代码块”时,其他线程仍然可以访问“该对象”的非同步代码块。 下面是“synchronized代码块”对应的演示程序。
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
41
42
43
44
45
46
47
48
49
50
51
52
53
54
class Count {
public void synMethod () {
synchronized (this ) {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName() + " synMethod loop " + i);
}
} catch (InterruptedException ie) {
}
}
}
public void nonSynMethod () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName() + " nonSynMethod loop " + i);
}
} catch (InterruptedException ie) {
}
}
}
public class Demo2 {
public static void main (String[] args) {
final Count count = new Count();
Thread t1 = new Thread(
new Runnable() {
@Override
public void run () {
count.synMethod();
}
}, "t1" );
Thread t2 = new Thread(
new Runnable() {
@Override
public void run () {
count.nonSynMethod();
}
}, "t2" );
t1.start();
t2.start();
}
}
运行结果:
t1 synMethod loop 0 t2 nonSynMethod loop 0 t1 synMethod loop 1 t2 nonSynMethod loop 1 t1 synMethod loop 2 t2 nonSynMethod loop 2 t1 synMethod loop 3 t2 nonSynMethod loop 3 t1 synMethod loop 4 t2 nonSynMethod loop 4
结果说明: 主线程中新建了两个子线程t1和t2。t1会调用count对象的synMethod()方法,该方法内含有同步块;而t2则会调用count对象的nonSynMethod()方法,该方法不是同步方法。t1运行时,虽然调用synchronized(this)获取“count的同步锁”;但是并没有造成t2的阻塞,因为t2没有用到“count”同步锁。
第三条 当一个线程访问“某对象”的“synchronized方法”或者“synchronized代码块”时,其他线程对“该对象”的其他的“synchronized方法”或者“synchronized代码块”的访问将被阻塞。 我们将上面的例子中的nonSynMethod()方法体的也用synchronized(this)修饰。修改后的源码如下:
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
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
class Count {
public void synMethod () {
synchronized (this ) {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName() + " synMethod loop " + i);
}
} catch (InterruptedException ie) {
}
}
}
public void nonSynMethod () {
synchronized (this ) {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName() + " nonSynMethod loop " + i);
}
} catch (InterruptedException ie) {
}
}
}
}
public class Demo3 {
public static void main (String[] args) {
final Count count = new Count();
Thread t1 = new Thread(
new Runnable() {
@Override
public void run () {
count.synMethod();
}
}, "t1" );
Thread t2 = new Thread(
new Runnable() {
@Override
public void run () {
count.nonSynMethod();
}
}, "t2" );
t1.start();
t2.start();
}
}
运行结果:
t1 synMethod loop 0 t1 synMethod loop 1 t1 synMethod loop 2 t1 synMethod loop 3 t1 synMethod loop 4 t2 nonSynMethod loop 0 t2 nonSynMethod loop 1 t2 nonSynMethod loop 2 t2 nonSynMethod loop 3 t2 nonSynMethod loop 4
结果说明: 主线程中新建了两个子线程t1和t2。t1和t2运行时都调用synchronized(this),这个this是Count对象(count),而t1和t2共用count。因此,在t1运行时,t2会被阻塞,等待t1运行释放“count对象的同步锁”,t2才能运行。
synchronized方法 和 synchronized代码块
“synchronized方法”是用synchronized修饰方法,而 “synchronized代码块”则是用synchronized修饰代码块。
synchronized方法示例
public synchronized void foo1() { System.out.println(“synchronized methoed”); } synchronized代码块
public void foo2() { synchronized (this) { System.out.println(“synchronized methoed”); } } synchronized代码块中的this是指当前对象。也可以将this替换成其他对象,例如将this替换成obj,则foo2()在执行synchronized(obj)时就获取的是obj的同步锁。
synchronized代码块可以更精确的控制冲突限制访问区域,有时候表现更高效率。下面通过一个示例来演示:
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
public class Demo4 {
public synchronized void synMethod () {
for (int i=0 ; i<1000000 ; i++)
;
}
public void synBlock () {
synchronized ( this ) {
for (int i=0 ; i<1000000 ; i++)
;
}
}
public static void main (String[] args) {
Demo4 demo = new Demo4();
long start, diff;
start = System.currentTimeMillis();
demo.synMethod();
diff = System.currentTimeMillis() - start;
System.out.println("synMethod() : " + diff);
start = System.currentTimeMillis();
demo.synBlock();
diff = System.currentTimeMillis() - start;
System.out.println("synBlock() : " + diff);
}
}
(某一次)执行结果:
synMethod() : 11 synBlock() : 3
实例锁 和 全局锁 实例锁 – 锁在某一个实例对象上。如果该类是单例,那么该锁也具有全局锁的概念。 实例锁对应的就是synchronized关键字。 全局锁 – 该锁针对的是类,无论实例多少个对象,那么线程都共享该锁。 全局锁对应的就是static synchronized(或者是锁在该类的class或者classloader对象上)。
关于“实例锁”和“全局锁”有一个很形象的例子:1
2
3
4
5
6
pulbic class Something {
public synchronized void isSyncA () {}
public synchronized void isSyncB () {}
public static synchronized void cSyncA () {}
public static synchronized void cSyncB () {}
}
假设,Something有两个实例x和y。分析下面4组表达式获取的锁的情况。 (01) x.isSyncA()与x.isSyncB() (02) x.isSyncA()与y.isSyncA() (03) x.cSyncA()与y.cSyncB() (04) x.isSyncA()与Something.cSyncA()
(01) 不能被同时访问。因为isSyncA()和isSyncB()都是访问同一个对象(对象x)的同步锁!
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
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
class Something {
public synchronized void isSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncA" );
}
}catch (InterruptedException ie) {
}
}
public synchronized void isSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncB" );
}
}catch (InterruptedException ie) {
}
}
}
public class LockTest1 {
Something x = new Something();
Something y = new Something();
private void test1 () {
Thread t11 = new Thread(
new Runnable() {
@Override
public void run () {
x.isSyncA();
}
}, "t11" );
Thread t12 = new Thread(
new Runnable() {
@Override
public void run () {
x.isSyncB();
}
}, "t12" );
t11.start();
t12.start();
}
public static void main (String[] args) {
LockTest1 demo = new LockTest1();
demo.test1();
}
}
运行结果:
t11 : isSyncA t11 : isSyncA t11 : isSyncA t11 : isSyncA t11 : isSyncA t12 : isSyncB t12 : isSyncB t12 : isSyncB t12 : isSyncB t12 : isSyncB
(02) 可以同时被访问。因为访问的不是同一个对象的同步锁,x.isSyncA()访问的是x的同步锁,而y.isSyncA()访问的是y的同步锁。
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
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
class Something {
public synchronized void isSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncA" );
}
}catch (InterruptedException ie) {
}
}
public synchronized void isSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncB" );
}
}catch (InterruptedException ie) {
}
}
public static synchronized void cSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : cSyncA" );
}
}catch (InterruptedException ie) {
}
}
public static synchronized void cSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : cSyncB" );
}
}catch (InterruptedException ie) {
}
}
}
public class LockTest2 {
Something x = new Something();
Something y = new Something();
private void test2 () {
Thread t21 = new Thread(
new Runnable() {
@Override
public void run () {
x.isSyncA();
}
}, "t21" );
Thread t22 = new Thread(
new Runnable() {
@Override
public void run () {
y.isSyncA();
}
}, "t22" );
t21.start();
t22.start();
}
public static void main (String[] args) {
LockTest2 demo = new LockTest2();
demo.test2();
}
}
运行结果:
t21 : isSyncA t22 : isSyncA t21 : isSyncA t22 : isSyncA t21 : isSyncA t22 : isSyncA t21 : isSyncA t22 : isSyncA t21 : isSyncA t22 : isSyncA
(03) 不能被同时访问。因为cSyncA()和cSyncB()都是static类型,x.cSyncA()相当于Something.isSyncA(),y.cSyncB()相当于Something.isSyncB(),因此它们共用一个同步锁,不能被同时反问。
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
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
class Something {
public synchronized void isSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncA" );
}
}catch (InterruptedException ie) {
}
}
public synchronized void isSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncB" );
}
}catch (InterruptedException ie) {
}
}
public static synchronized void cSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : cSyncA" );
}
}catch (InterruptedException ie) {
}
}
public static synchronized void cSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : cSyncB" );
}
}catch (InterruptedException ie) {
}
}
}
public class LockTest3 {
Something x = new Something();
Something y = new Something();
private void test3 () {
Thread t31 = new Thread(
new Runnable() {
@Override
public void run () {
x.cSyncA();
}
}, "t31" );
Thread t32 = new Thread(
new Runnable() {
@Override
public void run () {
y.cSyncB();
}
}, "t32" );
t31.start();
t32.start();
}
public static void main (String[] args) {
LockTest3 demo = new LockTest3();
demo.test3();
}
}
运行结果:
t31 : cSyncA t31 : cSyncA t31 : cSyncA t31 : cSyncA t31 : cSyncA t32 : cSyncB t32 : cSyncB t32 : cSyncB t32 : cSyncB t32 : cSyncB
(04) 可以被同时访问。因为isSyncA()是实例方法,x.isSyncA()使用的是对象x的锁;而cSyncA()是静态方法,Something.cSyncA()可以理解对使用的是“类的锁”。因此,它们是可以被同时访问的。
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
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
class Something {
public synchronized void isSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncA" );
}
}catch (InterruptedException ie) {
}
}
public synchronized void isSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : isSyncB" );
}
}catch (InterruptedException ie) {
}
}
public static synchronized void cSyncA () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : cSyncA" );
}
}catch (InterruptedException ie) {
}
}
public static synchronized void cSyncB () {
try {
for (int i = 0 ; i < 5 ; i++) {
Thread.sleep(100 );
System.out.println(Thread.currentThread().getName()+" : cSyncB" );
}
}catch (InterruptedException ie) {
}
}
}
public class LockTest4 {
Something x = new Something();
Something y = new Something();
private void test4 () {
Thread t41 = new Thread(
new Runnable() {
@Override
public void run () {
x.isSyncA();
}
}, "t41" );
Thread t42 = new Thread(
new Runnable() {
@Override
public void run () {
Something.cSyncA();
}
}, "t42" );
t41.start();
t42.start();
}
public static void main (String[] args) {
LockTest4 demo = new LockTest4();
demo.test4();
}
}
运行结果:
t41 : isSyncA t42 : cSyncA t41 : isSyncA t42 : cSyncA t41 : isSyncA t42 : cSyncA t41 : isSyncA t42 : cSyncA t41 : isSyncA t42 : cSyncA
wait(), notify(), notifyAll()等方法介绍 在Object.java中,定义了wait(), notify()和notifyAll()等接口。wait()的作用是让当前线程进入等待状态,同时,wait()也会让当前线程释放它所持有的锁。而notify()和notifyAll()的作用,则是唤醒当前对象上的等待线程;notify()是唤醒单个线程,而notifyAll()是唤醒所有的线程。
Object类中关于等待/唤醒的API详细信息如下: notify() – 唤醒在此对象监视器上等待的单个线程。
notifyAll() – 唤醒在此对象监视器上等待的所有线程。
wait() – 让当前线程处于“等待(阻塞)状态”,“直到其他线程调用此对象的 notify() 方法或 notifyAll() 方法”,当前线程被唤醒(进入“就绪状态”)。
wait(long timeout) – 让当前线程处于“等待(阻塞)状态”,“直到其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量”,当前线程被唤醒(进入“就绪状态”)。
wait(long timeout, int nanos) – 让当前线程处于“等待(阻塞)状态”,“直到其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者其他某个线程中断当前线程,或者已超过某个实际时间量”,当前线程被唤醒(进入“就绪状态”)。
wait()和notify()示例
下面通过示例演示”wait()和notify()配合使用的情形”。
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
class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
synchronized (this ) {
System.out.println(Thread.currentThread().getName()+" call notify()" );
notify();
}
}
}
public class WaitTest {
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
synchronized (t1) {
try {
System.out.println(Thread.currentThread().getName()+" start t1" );
t1.start();
System.out.println(Thread.currentThread().getName()+" wait()" );
t1.wait();
System.out.println(Thread.currentThread().getName()+" continue" );
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
运行结果:
main start t1 main wait() t1 call notify() main continue
结果说明: 如下图,说明了“主线程”和“线程t1”的流程。
(01) 注意,图中”主线程” 代表“主线程main”。”线程t1” 代表WaitTest中启动的“线程t1”。 而“锁” 代表“t1这个对象的同步锁”。 (02) “主线程”通过 new ThreadA(“t1”) 新建“线程t1”。随后通过synchronized(t1)获取“t1对象的同步锁”。然后调用t1.start()启动“线程t1”。 (03) “主线程”执行t1.wait() 释放“t1对象的锁”并且进入“等待(阻塞)状态”。等待t1对象上的线程通过notify() 或 notifyAll()将其唤醒。 (04) “线程t1”运行之后,通过synchronized(this)获取“当前对象的锁”;接着调用notify()唤醒“当前对象上的等待线程”,也就是唤醒“主线程”。 (05) “线程t1”运行完毕之后,释放“当前对象的锁”。紧接着,“主线程”获取“t1对象的锁”,然后接着运行。
对于上面的代码?曾经有个朋友问到过:t1.wait()应该是让“线程t1”等待;但是,为什么却是让“主线程main”等待了呢? 在解答该问题前,我们先看看jdk文档中关于wait的一段介绍:
Causes the current thread to wait until another thread invokes the notify() method or the notifyAll() method for this object. In other words, this method behaves exactly as if it simply performs the call wait(0). The current thread must own this object’s monitor. The thread releases ownership of this monitor and waits until another thread notifies threads waiting on this object’s monitor to wake up either through a call to the notify method or the notifyAll method. The thread then waits until it can re-obtain ownership of the monitor and resumes execution.
中文意思大概是:
引起“当前线程”等待,直到另外一个线程调用notify()或notifyAll()唤醒该线程。换句话说,这个方法和wait(0)的效果一样!(补充,对于wait(long millis)方法,当millis为0时,表示无限等待,直到被notify()或notifyAll()唤醒)。 “当前线程”在调用wait()时,必须拥有该对象的同步锁。该线程调用wait()之后,会释放该锁;然后一直等待直到“其它线程”调用对象的同步锁的notify()或notifyAll()方法。然后,该线程继续等待直到它重新获取“该对象的同步锁”,然后就可以接着运行。
注意:jdk的解释中,说wait()的作用是让“当前线程”等待,而“当前线程”是指正在cpu上运行的线程! 这也意味着,虽然t1.wait()是通过“线程t1”调用的wait()方法,但是调用t1.wait()的地方是在“主线程main”中。而主线程必须是“当前线程”,也就是运行状态,才可以执行t1.wait()。所以,此时的“当前线程”是“主线程main”!因此,t1.wait()是让“主线程”等待,而不是“线程t1”!
wait(long timeout)和notify()
wait(long timeout)会让当前线程处于“等待(阻塞)状态”,“直到其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量”,当前线程被唤醒(进入“就绪状态”)。 下面的示例就是演示wait(long timeout)在超时情况下,线程被唤醒的情况。
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
class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
System.out.println(Thread.currentThread().getName() + " run " );
while (true )
;
}
}
public class WaitTimeoutTest {
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
synchronized (t1) {
try {
System.out.println(Thread.currentThread().getName() + " start t1" );
t1.start();
System.out.println(Thread.currentThread().getName() + " call wait " );
t1.wait(3000 );
System.out.println(Thread.currentThread().getName() + " continue" );
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
运行结果:
main start t1 main call wait t1 run // 大约3秒之后…输出“main continue” main continue
结果说明: 如下图,说明了“主线程”和“线程t1”的流程。 (01) 注意,图中”主线程” 代表WaitTimeoutTest主线程(即,线程main)。”线程t1” 代表WaitTest中启动的线程t1。 而“锁” 代表“t1这个对象的同步锁”。 (02) 主线程main执行t1.start()启动“线程t1”。 (03) 主线程main执行t1.wait(3000),此时,主线程进入“阻塞状态”。需要“用于t1对象锁的线程通过notify() 或者 notifyAll()将其唤醒” 或者 “超时3000ms之后”,主线程main才进入到“就绪状态”,然后才可以运行。 (04) “线程t1”运行之后,进入了死循环,一直不断的运行。 (05) 超时3000ms之后,主线程main会进入到“就绪状态”,然后接着进入“运行状态”。
wait() 和 notifyAll() 通过前面的示例,我们知道 notify() 可以唤醒在此对象监视器上等待的单个线程。 下面,我们通过示例演示notifyAll()的用法;它的作用是唤醒在此对象监视器上等待的所有线程。
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
41
42
43
44
45
46
47
48
49
50
public class NotifyAllTest {
private static Object obj = new Object();
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
ThreadA t2 = new ThreadA("t2" );
ThreadA t3 = new ThreadA("t3" );
t1.start();
t2.start();
t3.start();
try {
System.out.println(Thread.currentThread().getName()+" sleep(3000)" );
Thread.sleep(3000 );
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (obj) {
System.out.println(Thread.currentThread().getName()+" notifyAll()" );
obj.notifyAll();
}
}
static class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
synchronized (obj) {
try {
System.out.println(Thread.currentThread().getName() + " wait" );
obj.wait();
System.out.println(Thread.currentThread().getName() + " continue" );
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
运行结果:
t1 wait main sleep(3000) t3 wait t2 wait main notifyAll() t2 continue t3 continue t1 continue
结果说明: 参考下面的流程图。
(01) 主线程中新建并且启动了3个线程”t1”, “t2”和”t3”。 (02) 主线程通过sleep(3000)休眠3秒。在主线程休眠3秒的过程中,我们假设”t1”, “t2”和”t3”这3个线程都运行了。以”t1”为例,当它运行的时候,它会执行obj.wait()等待其它线程通过notify()或额nofityAll()来唤醒它;相同的道理,”t2”和”t3”也会等待其它线程通过nofity()或nofityAll()来唤醒它们。 (03) 主线程休眠3秒之后,接着运行。执行 obj.notifyAll() 唤醒obj上的等待线程,即唤醒”t1”, “t2”和”t3”这3个线程。 紧接着,主线程的synchronized(obj)运行完毕之后,主线程释放“obj锁”。这样,”t1”, “t2”和”t3”就可以获取“obj锁”而继续运行了!
为什么notify(), wait()等函数定义在Object中,而不是Thread中
Object中的wait(), notify()等函数,和synchronized一样,会对“对象的同步锁”进行操作。
wait()会使“当前线程”等待,因为线程进入等待状态,所以线程应该释放它锁持有的“同步锁”,否则其它线程获取不到该“同步锁”而无法运行! OK,线程调用wait()之后,会释放它锁持有的“同步锁”;而且,根据前面的介绍,我们知道:等待线程可以被notify()或notifyAll()唤醒。现在,请思考一个问题:notify()是依据什么唤醒等待线程的?或者说,wait()等待线程和notify()之间是通过什么关联起来的?答案是:依据“对象的同步锁”。
负责唤醒等待线程的那个线程(我们称为“唤醒线程”),它只有在获取“该对象的同步锁”(这里的同步锁必须和等待线程的同步锁是同一个),并且调用notify()或notifyAll()方法之后,才能唤醒等待线程。虽然,等待线程被唤醒;但是,它不能立刻执行,因为唤醒线程还持有“该对象的同步锁”。必须等到唤醒线程释放了“对象的同步锁”之后,等待线程才能获取到“对象的同步锁”进而继续运行。
总之,notify(), wait()依赖于“同步锁”,而“同步锁”是对象锁持有,并且每个对象有且仅有一个!这就是为什么notify(), wait()等函数定义在Object类,而不是Thread类中的原因。
线程让步
yield()介绍 yield()的作用是让步。它能让当前线程由“运行状态”进入到“就绪状态”,从而让其它具有相同优先级的等待线程获取执行权;但是,并不能保证在当前线程调用yield()之后,其它具有相同优先级的线程就一定能获得执行权;也有可能是当前线程又进入到“运行状态”继续运行!
yield()示例 下面,通过示例查看它的用法。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public synchronized void run () {
for (int i=0 ; i <10 ; i++){
System.out.printf("%s [%d]:%d\n" , this .getName(), this .getPriority(), i);
if (i%4 == 0 )
Thread.yield();
}
}
}
public class YieldTest {
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
ThreadA t2 = new ThreadA("t2" );
t1.start();
t2.start();
}
}
(某一次的)运行结果:
t1 [5]:0 t2 [5]:0 t1 [5]:1 t1 [5]:2 t1 [5]:3 t1 [5]:4 t1 [5]:5 t1 [5]:6 t1 [5]:7 t1 [5]:8 t1 [5]:9 t2 [5]:1 t2 [5]:2 t2 [5]:3 t2 [5]:4 t2 [5]:5 t2 [5]:6 t2 [5]:7 t2 [5]:8 t2 [5]:9
结果说明: “线程t1”在能被4整数的时候,并没有切换到“线程t2”。这表明,yield()虽然可以让线程由“运行状态”进入到“就绪状态”;但是,它不一定会让其它线程获取CPU执行权(即,其它线程进入到“运行状态”),即使这个“其它线程”与当前调用yield()的线程具有相同的优先级。
yield() 与 wait()的比较
我们知道,wait()的作用是让当前线程由“运行状态”进入“等待(阻塞)状态”的同时,也会释放同步锁。而yield()的作用是让步,它也会让当前线程离开“运行状态”。它们的区别是: (01) wait()是让线程由“运行状态”进入到“等待(阻塞)状态”,而不yield()是让线程由“运行状态”进入到“就绪状态”。 (02) wait()是会线程释放它所持有对象的同步锁,而yield()方法不会释放锁。
下面通过示例演示yield()是不会释放锁的。
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
public class YieldLockTest {
private static Object obj = new Object();
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
ThreadA t2 = new ThreadA("t2" );
t1.start();
t2.start();
}
static class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
synchronized (obj) {
for (int i=0 ; i <10 ; i++){
System.out.printf("%s [%d]:%d\n" , this .getName(), this .getPriority(), i);
if (i%4 == 0 )
Thread.yield();
}
}
}
}
}
(某一次)运行结果:
t1 [5]:0 t1 [5]:1 t1 [5]:2 t1 [5]:3 t1 [5]:4 t1 [5]:5 t1 [5]:6 t1 [5]:7 t1 [5]:8 t1 [5]:9 t2 [5]:0 t2 [5]:1 t2 [5]:2 t2 [5]:3 t2 [5]:4 t2 [5]:5 t2 [5]:6 t2 [5]:7 t2 [5]:8 t2 [5]:9
结果说明: 主线程main中启动了两个线程t1和t2。t1和t2在run()会引用同一个对象的同步锁,即synchronized(obj)。在t1运行过程中,虽然它会调用Thread.yield();但是,t2是不会获取cpu执行权的。因为,t1并没有释放“obj所持有的同步锁”!
sleep
sleep()介绍 sleep() 定义在Thread.java中。 sleep() 的作用是让当前线程休眠,即当前线程会从“运行状态”进入到“休眠(阻塞)状态”。sleep()会指定休眠时间,线程休眠的时间会大于/等于该休眠时间;在线程重新被唤醒时,它会由“阻塞状态”变成“就绪状态”,从而等待cpu的调度执行。
sleep()示例 下面通过一个简单示例演示sleep()的用法。
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
class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public synchronized void run () {
try {
for (int i=0 ; i <10 ; i++){
System.out.printf("%s: %d\n" , this .getName(), i);
if (i%4 == 0 )
Thread.sleep(100 );
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public class SleepTest {
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
t1.start();
}
}
运行结果:
t1: 0 t1: 1 t1: 2 t1: 3 t1: 4 t1: 5 t1: 6 t1: 7 t1: 8 t1: 9
结果说明: 程序比较简单,在主线程main中启动线程t1。t1启动之后,当t1中的计算i能被4整除时,t1会通过Thread.sleep(100)休眠100毫秒。
sleep() 与 wait()的比较 我们知道,wait()的作用是让当前线程由“运行状态”进入“等待(阻塞)状态”的同时,也会释放同步锁。而sleep()的作用是也是让当前线程由“运行状态”进入到“休眠(阻塞)状态”。 但是,wait()会释放对象的同步锁,而sleep()则不会释放锁。 下面通过示例演示sleep()是不会释放锁的。
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
public class SleepLockTest {
private static Object obj = new Object();
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
ThreadA t2 = new ThreadA("t2" );
t1.start();
t2.start();
}
static class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
synchronized (obj) {
try {
for (int i=0 ; i <10 ; i++){
System.out.printf("%s: %d\n" , this .getName(), i);
if (i%4 == 0 )
Thread.sleep(100 );
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
运行结果:
t1: 0 t1: 1 t1: 2 t1: 3 t1: 4 t1: 5 t1: 6 t1: 7 t1: 8 t1: 9 t2: 0 t2: 1 t2: 2 t2: 3 t2: 4 t2: 5 t2: 6 t2: 7 t2: 8 t2: 9
结果说明: 主线程main中启动了两个线程t1和t2。t1和t2在run()会引用同一个对象的同步锁,即synchronized(obj)。在t1运行过程中,虽然它会调用Thread.sleep(100);但是,t2是不会获取cpu执行权的。因为,t1并没有释放“obj所持有的同步锁”! 注意,若我们注释掉synchronized (obj)后再次执行该程序,t1和t2是可以相互切换的。下面是注释调synchronized(obj) 之后的源码:
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
public class SleepLockTest {
private static Object obj = new Object();
public static void main (String[] args) {
ThreadA t1 = new ThreadA("t1" );
ThreadA t2 = new ThreadA("t2" );
t1.start();
t2.start();
}
static class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
try {
for (int i=0 ; i <10 ; i++){
System.out.printf("%s: %d\n" , this .getName(), i);
if (i%4 == 0 )
Thread.sleep(100 );
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
join
join()介绍 join() 定义在Thread.java中。 join() 的作用:让“主线程”等待“子线程”结束之后才能继续运行。这句话可能有点晦涩,我们还是通过例子去理解:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public class Father extends Thread {
public void run () {
Son s = new Son();
s.start();
s.join();
...
}
}
public class Son extends Thread {
public void run () {
...
}
}
说明: 上面的有两个类Father(主线程类)和Son(子线程类)。因为Son是在Father中创建并启动的,所以,Father是主线程类,Son是子线程类。 在Father主线程中,通过new Son()新建“子线程s”。接着通过s.start()启动“子线程s”,并且调用s.join()。在调用s.join()之后,Father主线程会一直等待,直到“子线程s”运行完毕;在“子线程s”运行完毕之后,Father主线程才能接着运行。 这也就是我们所说的“join()的作用,是让主线程会等待子线程结束之后才能继续运行”!
join()源码分析(基于JDK1.7.0_40)
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
public final void join () throws InterruptedException {
join(0 );
}
public final synchronized void join (long millis)
throws InterruptedException {
long base = System.currentTimeMillis();
long now = 0 ;
if (millis < 0 ) {
throw new IllegalArgumentException("timeout value is negative" );
}
if (millis == 0 ) {
while (isAlive()) {
wait(0 );
}
} else {
while (isAlive()) {
long delay = millis - now;
if (delay <= 0 ) {
break ;
}
wait(delay);
now = System.currentTimeMillis() - base;
}
}
}
说明: 从代码中,我们可以发现。当millis==0时,会进入while(isAlive())循环;即只要子线程是活的,主线程就不停的等待。 我们根据上面解释join()作用时的代码来理解join()的用法! 问题: 虽然s.join()被调用的地方是发生在“Father主线程”中,但是s.join()是通过“子线程s”去调用的join()。那么,join()方法中的isAlive()应该是判断“子线程s”是不是Alive状态;对应的wait(0)也应该是“让子线程s”等待才对。但如果是这样的话,s.join()的作用怎么可能是“让主线程等待,直到子线程s完成为止”呢,应该是让”子线程等待才对(因为调用子线程对象s的wait方法嘛)”? 答案:wait()的作用是让“当前线程”等待,而这里的“当前线程”是指当前在CPU上运行的线程。所以,虽然是调用子线程的wait()方法,但是它是通过“主线程”去调用的;所以,休眠的是主线程,而不是“子线程”!
join()示例
在理解join()的作用之后,接下来通过示例查看join()的用法。
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
public class JoinTest {
public static void main (String[] args) {
try {
ThreadA t1 = new ThreadA("t1" );
t1.start();
t1.join();
System.out.printf("%s finish\n" , Thread.currentThread().getName());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
static class ThreadA extends Thread {
public ThreadA (String name) {
super (name);
}
public void run () {
System.out.printf("%s start\n" , this .getName());
for (int i=0 ; i <1000000 ; i++)
;
System.out.printf("%s finish\n" , this .getName());
}
}
}
运行结果:
t1 start t1 finish main finish 结果说明: 运行流程如图 (01) 在“主线程main”中通过 new ThreadA(“t1”) 新建“线程t1”。 接着,通过 t1.start() 启动“线程t1”,并执行t1.join()。 (02) 执行t1.join()之后,“主线程main”会进入“阻塞状态”等待t1运行结束。“子线程t1”结束之后,会唤醒“主线程main”,“主线程”重新获取cpu执行权,继续运行。
interrupt
interrupt()说明 在介绍终止线程的方式之前,有必要先对interrupt()进行了解。 关于interrupt(),java的djk文档描述如下:http://docs.oracle.com/javase/7/docs/api/
1
2
3
4
5
6
7
8
9
10
11
12
Interrupts this thread.
Unless the current thread is interrupting itself, which is always permitted, the checkAccess method of this thread is invoked, which may cause a SecurityException to be thrown.
If this thread is blocked in an invocation of the wait () , wait (long ) , or wait (long , int ) methods of the Object class, or of the join () , join (long ) , join (long , int ) , sleep (long ) , or sleep (long , int ) , methods of this class, then its interrupt status will be cleared and it will receive an InterruptedException.
If this thread is blocked in an I/O operation upon an interruptible channel then the channel will be closed, the thread's interrupt status will be set, and the thread will receive a ClosedByInterruptException.
If this thread is blocked in a Selector then the thread's interrupt status will be set and it will return immediately from the selection operation, possibly with a non-zero value, just as if the selector's wakeup method were invoked.
If none of the previous conditions hold then this thread's interrupt status will be set.
Interrupting a thread that is not alive need not have any effect.
大致意思是:
1
2
3
4
5
6
interrupt()的作用是中断本线程。
本线程中断自己是被允许的;其它线程调用本线程的interrupt()方法时,会通过checkAccess()检查权限。这有可能抛出SecurityException异常。
如果本线程是处于阻塞状态:调用线程的wait(), wait(long )或wait(long , int )会让它进入等待(阻塞)状态,或者调用线程的join(), join(long ), join(long , int ), sleep(long ), sleep(long , int )也会让它进入阻塞状态。若线程在阻塞状态时,调用了它的interrupt()方法,那么它的“中断状态”会被清除并且会收到一个InterruptedException异常。例如,线程通过wait()进入阻塞状态,此时通过interrupt()中断该线程;调用interrupt()会立即将线程的中断标记设为“true ”,但是由于线程处于阻塞状态,所以该“中断标记”会立即被清除为“false ”,同时,会产生一个InterruptedException的异常。
如果线程被阻塞在一个Selector选择器中,那么通过interrupt()中断它时;线程的中断标记会被设置为true ,并且它会立即从选择操作中返回。
如果不属于前面所说的情况,那么通过interrupt()中断线程时,它的中断标记会被设置为“true ”。
中断一个“已终止的线程”不会产生任何操作。
终止线程的方式 Thread中的stop()和suspend()方法,由于固有的不安全性,已经建议不再使用! 下面,我先分别讨论线程在“阻塞状态”和“运行状态”的终止方式,然后再总结出一个通用的方式。
1 终止处于“阻塞状态”的线程 通常,我们通过“中断”方式终止处于“阻塞状态”的线程。 当线程由于被调用了sleep(), wait(), join()等方法而进入阻塞状态;若此时调用线程的interrupt()将线程的中断标记设为true。由于处于阻塞状态,中断标记会被清除,同时产生一个InterruptedException异常。将InterruptedException放在适当的为止就能终止线程,形式如下:
1
2
3
4
5
6
7
8
9
10
@Override
public void run () {
try {
while (true ) {
}
} catch (InterruptedException ie) {
}
}
说明:在while(true)中不断的执行任务,当线程处于阻塞状态时,调用线程的interrupt()产生InterruptedException中断。中断的捕获在while(true)之外,这样就退出了while(true)循环! 注意:对InterruptedException的捕获务一般放在while(true)循环体的外面,这样,在产生异常时就退出了while(true)循环。否则,InterruptedException在while(true)循环体之内,就需要额外的添加退出处理。形式如下:
1
2
3
4
5
6
7
8
9
10
11
12
@Override
public void run () {
while (true ) {
try {
} catch (InterruptedException ie) {
break ;
}
}
}
说明:上面的InterruptedException异常的捕获在whle(true)之内。当产生InterruptedException异常时,被catch处理之外,仍然在while(true)循环体内;要退出while(true)循环体,需要额外的执行退出while(true)的操作。
终止处于“运行状态”的线程 通常,我们通过“标记”方式终止处于“运行状态”的线程。其中,包括“中断标记”和“额外添加标记”。 (01) 通过“中断标记”终止线程。 形式如下:1
2
3
4
5
6
@Override
public void run () {
while (!isInterrupted()) {
}
}
说明:isInterrupted()是判断线程的中断标记是不是为true。当线程处于运行状态,并且我们需要终止它时;可以调用线程的interrupt()方法,使用线程的中断标记为true,即isInterrupted()会返回true。此时,就会退出while循环。 注意:interrupt()并不会终止处于“运行状态”的线程!它会将线程的中断标记设为true。
(02) 通过“额外添加标记”。 形式如下:
1
2
3
4
5
6
7
8
9
10
11
private volatile boolean flag= true ;
protected void stopTask () {
flag = false ;
}
@Override
public void run () {
while (flag) {
}
}
说明:线程中有一个flag标记,它的默认值是true;并且我们提供stopTask()来设置flag标记。当我们需要终止该线程时,调用该线程的stopTask()方法就可以让线程退出while循环。 注意:将flag定义为volatile类型,是为了保证flag的可见性。即其它线程通过stopTask()修改了flag之后,本线程能看到修改后的flag的值。
综合线程处于“阻塞状态”和“运行状态”的终止方式,比较通用的终止线程的形式如下:
1
2
3
4
5
6
7
8
9
10
11
@Override
public void run () {
try {
while (!isInterrupted()) {
}
} catch (InterruptedException ie) {
}
}
终止线程的示例
interrupt()常常被用来终止“阻塞状态”线程。参考下面示例:
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
41
42
43
44
45
class MyThread extends Thread {
public MyThread (String name) {
super (name);
}
@Override
public void run () {
try {
int i=0 ;
while (!isInterrupted()) {
Thread.sleep(100 );
i++;
System.out.println(Thread.currentThread().getName()+" (" +this .getState()+") loop " + i);
}
} catch (InterruptedException e) {
System.out.println(Thread.currentThread().getName() +" (" +this .getState()+") catch InterruptedException." );
}
}
}
public class Demo1 {
public static void main (String[] args) {
try {
Thread t1 = new MyThread("t1" );
System.out.println(t1.getName() +" (" +t1.getState()+") is new." );
t1.start();
System.out.println(t1.getName() +" (" +t1.getState()+") is started." );
Thread.sleep(300 );
t1.interrupt();
System.out.println(t1.getName() +" (" +t1.getState()+") is interrupted." );
Thread.sleep(300 );
System.out.println(t1.getName() +" (" +t1.getState()+") is interrupted now." );
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
运行结果:
t1 (NEW) is new. t1 (RUNNABLE) is started. t1 (RUNNABLE) loop 1 t1 (RUNNABLE) loop 2 t1 (TIMED_WAITING) is interrupted. t1 (RUNNABLE) catch InterruptedException. t1 (TERMINATED) is interrupted now.
结果说明: (01) 主线程main中通过new MyThread(“t1”)创建线程t1,之后通过t1.start()启动线程t1。 (02) t1启动之后,会不断的检查它的中断标记,如果中断标记为“false”;则休眠100ms。 (03) t1休眠之后,会切换到主线程main;主线程再次运行时,会执行t1.interrupt()中断线程t1。t1收到中断指令之后,会将t1的中断标记设置“false”,而且会抛出InterruptedException异常。在t1的run()方法中,是在循环体while之外捕获的异常;因此循环被终止。
我们对上面的结果进行小小的修改,将run()方法中捕获InterruptedException异常的代码块移到while循环体内。
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
41
42
43
44
45
class MyThread extends Thread {
public MyThread (String name) {
super (name);
}
@Override
public void run () {
int i=0 ;
while (!isInterrupted()) {
try {
Thread.sleep(100 );
} catch (InterruptedException ie) {
System.out.println(Thread.currentThread().getName() +" (" +this .getState()+") catch InterruptedException." );
}
i++;
System.out.println(Thread.currentThread().getName()+" (" +this .getState()+") loop " + i);
}
}
}
public class Demo2 {
public static void main (String[] args) {
try {
Thread t1 = new MyThread("t1" );
System.out.println(t1.getName() +" (" +t1.getState()+") is new." );
t1.start();
System.out.println(t1.getName() +" (" +t1.getState()+") is started." );
Thread.sleep(300 );
t1.interrupt();
System.out.println(t1.getName() +" (" +t1.getState()+") is interrupted." );
Thread.sleep(300 );
System.out.println(t1.getName() +" (" +t1.getState()+") is interrupted now." );
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
运行结果:
t1 (NEW) is new. t1 (RUNNABLE) is started. t1 (RUNNABLE) loop 1 t1 (RUNNABLE) loop 2 t1 (TIMED_WAITING) is interrupted. t1 (RUNNABLE) catch InterruptedException. t1 (RUNNABLE) loop 3 t1 (RUNNABLE) loop 4 t1 (RUNNABLE) loop 5 t1 (TIMED_WAITING) is interrupted now. t1 (RUNNABLE) loop 6 t1 (RUNNABLE) loop 7 t1 (RUNNABLE) loop 8 t1 (RUNNABLE) loop 9 …
结果说明: 程序进入了死循环! 为什么会这样呢?这是因为,t1在“等待(阻塞)状态”时,被interrupt()中断;此时,会清除中断标记[即isInterrupted()会返回false],而且会抛出InterruptedException异常[该异常在while循环体内被捕获]。因此,t1理所当然的会进入死循环了。 解决该问题,需要我们在捕获异常时,额外的进行退出while循环的处理。例如,在MyThread的catch(InterruptedException)中添加break 或 return就能解决该问题。
下面是通过“额外添加标记”的方式终止“状态状态”的线程的示例:
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
41
42
43
44
45
46
47
48
49
50
51
52
class MyThread extends Thread {
private volatile boolean flag= true ;
public void stopTask () {
flag = false ;
}
public MyThread (String name) {
super (name);
}
@Override
public void run () {
synchronized (this ) {
try {
int i=0 ;
while (flag) {
Thread.sleep(100 );
i++;
System.out.println(Thread.currentThread().getName()+" (" +this .getState()+") loop " + i);
}
} catch (InterruptedException ie) {
System.out.println(Thread.currentThread().getName() +" (" +this .getState()+") catch InterruptedException." );
}
}
}
}
public class Demo3 {
public static void main (String[] args) {
try {
MyThread t1 = new MyThread("t1" );
System.out.println(t1.getName() +" (" +t1.getState()+") is new." );
t1.start();
System.out.println(t1.getName() +" (" +t1.getState()+") is started." );
Thread.sleep(300 );
t1.stopTask();
System.out.println(t1.getName() +" (" +t1.getState()+") is interrupted." );
Thread.sleep(300 );
System.out.println(t1.getName() +" (" +t1.getState()+") is interrupted now." );
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
运行结果:
t1 (NEW) is new. t1 (RUNNABLE) is started. t1 (RUNNABLE) loop 1 t1 (RUNNABLE) loop 2 t1 (TIMED_WAITING) is interrupted. t1 (RUNNABLE) loop 3 t1 (TERMINATED) is interrupted now.
interrupted() 和 isInterrupted()的区别
最后谈谈 interrupted() 和 isInterrupted()。 interrupted() 和 isInterrupted()都能够用于检测对象的“中断标记”。 区别是,interrupted()除了返回中断标记之外,它还会清除中断标记(即将中断标记设为false);而isInterrupted()仅仅返回中断标记。