1.Java多线程之wait和notify

1.首先我们来从概念上理解一下这两个方法:

(1)obj.wait(),当obj对象调用wait方法时,这个方法会让当前执行了这条语句的线程处于等待状态(或者说阻塞状态),并释放调用wait方法的对象的对象锁。

当线程执行了obj.wait()这个语句之后,这个线程处于等待状态,需要其它线程用同一个obj对象调用notify或notifyAll方法,才可能唤醒这个处于等待状态的的线程。

(2)obj.notify(), 调用这个方法,就是唤醒其它 在obj这对象锁上处于等待状态 的线程,被唤醒的线程 在得到对象锁后 就会继续执行。

 

概念上的文字晦涩、让人理解的不够清晰,

我们来看代码吧,

这个小程序的目的就是:启两个线程,并让这两个线程切换着执行,并按序打印ababababababababababab。

 1 package cn.javaBase.study_thread1;
 2 
 3 
 4 class MyRunnable implements Runnable{
 5     private String name;
 6     private Object obj;
 7     
 8     public MyRunnable(String n, Object o) {
 9         this.name = n;
10         this.obj = o;
11     }
12     
13     @Override
14     public void run() {
15         synchronized (obj) {
16             while (true) {
17                 System.out.println(name);
18                 obj.notify();  //当a线程执行到这里时,它就会唤醒另一个在obj对象锁中 处于等待状态的线程
19                                 //(也就是另一个在obj对象锁中调用了obj.wait() 语句的线程)
20                                 //注意,这里它不是唤醒 所有等待线程 中的任意一个, 而是唤醒 在obj这个对象锁上处于等待状态的 线程
21                 try {
22                     obj.wait();  //当a线程执行到这里,它就会让 当前线程a处于等待状态,并释放obj这个对象。
23                     System.out.println("当前线程,会阻塞在这里");
24                 } catch (InterruptedException e) {
25                     e.printStackTrace();
26                 }
27             }
28         }
29     }
30     
31 }
32 
33 public class Thread5AB {
34 
35     public static void main(String[] args) throws InterruptedException {
36         //这个程序中 共同的锁对象
37         Object o = new Object();
38         
39         MyRunnable r1 = new MyRunnable("a", o);
40         MyRunnable r2 = new MyRunnable("b", o);
41         
42         Thread th1 = new Thread(r1);
43         Thread th2 = new Thread(r2);
44         
45         th1.start();
46         Thread.sleep(500); //这个是保证打印a的线程先启动
47         th2.start();
48         
49     }
50 }

为了描述方便, 假设 上面的th1 和th2 分别代表着a、b线程

对于这个程序,重点应该是理解39、40行,r1和r2构造时,会什么传递的是同一个Object 类型的o对象,

因为:当a线程执行到22行时,a线程就处于等待状态,这时候b线程执行,当b线程直线到18行时,b线程就可以唤醒阻塞在o对象锁上的线程,而此时a线程不就是那个阻塞在o对象锁上的线程吗,

所以等b线程执行到18行时,a线程就被唤醒(但是这个时候a线程还不能进入synchronized代码块,就是这个对象锁中,因为进入这个代码块,需要有obj这个对象,obj这个对象就像是进入这个代码块的钥匙),

接下来等到b线程执行到22行时,b线程就把obj这个对象释放了,自己进入等待状态,此时a线程已经处于唤醒了的状态,并可以得到obj这个对象,所以接下来a线程就可以继续执行synchronized代码块中的内容,

这就是a、b线程第一次切换执行的过程,后面就依次这样循环执行,

控制台就打印了我们想要的结果ababababab...

重要的问题是,你要知道obj.notify()执行时,唤醒的是那个处于等待的线程,它唤醒的是在obj这个对象锁上处于等待的线程

不知道这样解释的是否还可以,本人也是最近稿明白,

posted @ 2018-04-10 17:35  zailushan  阅读(270)  评论(0编辑  收藏  举报