【转】通过生产者消费者案例理解等待唤醒机制和虚假唤醒

首先引入下面这段生产者和消费者的程序,店员类作为生产产品和消费产品的中介,其中的数据product为共享数据,产品最多只能囤积5个,当产品达到5个还在生产时,就会提示“产品已满!”,类似地,如果产品只有0个了还在消费,会提示“缺货!”:

 1 package concurrent;
 2
 3 //店员类
 4 class Clerk {
 5     private int product = 0;
 6
 7     // 进货
 8     public synchronized void get() {
 9         if (product >= 5) {
10             System.out.println("产品已满!");
11         } else {
12             System.out.println(Thread.currentThread().getName() + ":" + ++product);
13         }
14     }
15
16     // 售货
17     public synchronized void sale() {
18         if (product <= 0) {
19             System.out.println("缺货!");
20         } else {
21             System.out.println(Thread.currentThread().getName() + ":" + --product);
22         }
23     }
24 }
25
26 // 生产者类
27 class Productor implements Runnable {
28
29     private Clerk clerk;
30
31     public Productor(Clerk clerk) {
32         this.clerk = clerk;
33     }
34
35     @Override
36     public void run() {
37         for (int i = 0; i < 10; i++) {
38             clerk.get();
39         }
40
41     }
42 }
43
44 //消费者类
45 class Consumer implements Runnable {
46
47     private Clerk clerk;
48
49     public Consumer(Clerk clerk) {
50         this.clerk = clerk;
51     }
52
53     @Override
54     public void run() {
55         for (int i = 0; i < 10; i++) {
56             clerk.sale();
57         }
58     }
59 }
60
61 public class TestProductorAndConsumer {
62
63     public static void main(String[] args) {
64         Clerk clerk = new Clerk();
65
66         Productor productor = new Productor(clerk);
67         Consumer consumer = new Consumer(clerk);
68
69         new Thread(productor,"Productor A").start();
70         new Thread(consumer,"Consumer B").start();
71     }
72 }

运行程序,结果如下:

这是一种不好的情况,因为当产品已满时,还在不停地生产,当缺货时,还在不停地消费。为此,我们引入等待唤醒机制:

 1 package concurrent;
 2
 3 //店员类
 4 class Clerk {
 5     private int product = 0;
 6
 7     // 进货
 8     public synchronized void get() {
 9         if (product >= 5) {
10             System.out.println("产品已满!");
11
12             //等待
13             try {
14                 this.wait();
15             } catch (InterruptedException e) {
16                 e.printStackTrace();
17             }
18         } else {
19             System.out.println(Thread.currentThread().getName() + ":" + ++product);
20             //唤醒
21             this.notifyAll();
22         }
23     }
24
25     // 售货
26     public synchronized void sale() {
27         if (product <= 0) {
28             System.out.println("缺货!");
29             //等待
30             try {
31                 this.wait();
32             } catch (InterruptedException e) {
33                 e.printStackTrace();
34             }
35         } else {
36             System.out.println(Thread.currentThread().getName() + ":" + --product);
37             //唤醒
38             this.notifyAll();
39         }
40     }
41 }
42
43 // 生产者类
44 class Productor implements Runnable {
45
46     private Clerk clerk;
47
48     public Productor(Clerk clerk) {
49         this.clerk = clerk;
50     }
51
52     @Override
53     public void run() {
54         for (int i = 0; i < 10; i++) {
55             clerk.get();
56         }
57     }
58 }
59
60 //消费者类
61 class Consumer implements Runnable {
62
63     private Clerk clerk;
64
65     public Consumer(Clerk clerk) {
66         this.clerk = clerk;
67     }
68
69     @Override
70     public void run() {
71         for (int i = 0; i < 10; i++) {
72             clerk.sale();
73         }
74
75     }
76 }
77
78 public class TestProductorAndConsumer {
79
80     public static void main(String[] args) {
81         Clerk clerk = new Clerk();
82
83         Productor productor = new Productor(clerk);
84         Consumer consumer = new Consumer(clerk);
85
86         new Thread(productor,"Productor A").start();
87         new Thread(consumer,"Consumer B").start();
88     }
89 }

再运行程序,就不会再出现上述的情况:

但是,现在,我们将产品的囤积上限设定为1(这种情况在现实中也是有可能出现的):

然后运行程序:

程序的输出貌似没有问题,但请注意图中箭头所指的地方,这表示程序没有结束,还一直在执行。这是因为,当循坏到最后一轮时,由于产品已满引发了wait()操作,然后生产者线程等待,随后消费者消费了一份产品,并唤醒等待的生产者线程,此时,被唤醒的生产者线程由于循环结束,直接结束了线程的执行,但是另一边,消费者线程没有结束,而且由于将产品消费完后再次进入了等待,但是生产者线程此时已经结束了,不能再唤醒消费者线程,所以便进入了死循环。

解决这种问题的方法时去掉Clerk类中get方法和sale方法的else,并将原来else中的代码直接提出,这样,就算线程结束,也会先再次唤醒等待的线程:

 1 package concurrent;
 2
 3 //店员类
 4 class Clerk {
 5     private int product = 0;
 6
 7     // 进货
 8     public synchronized void get() {
 9         if (product >= 1) {
10             System.out.println("产品已满!");
11
12             // 等待
13             try {
14                 this.wait();
15             } catch (InterruptedException e) {
16                 e.printStackTrace();
17             }
18         }
19         System.out.println(Thread.currentThread().getName() + ":" + ++product);
20         // 唤醒
21         this.notifyAll();
22     }
23
24     // 售货
25     public synchronized void sale() {
26         if (product <= 0) {
27             System.out.println("缺货!");
28             // 等待
29             try {
30                 this.wait();
31             } catch (InterruptedException e) {
32                 e.printStackTrace();
33             }
34         }
35         System.out.println(Thread.currentThread().getName() + ":" + --product);
36         // 唤醒
37         this.notifyAll();
38     }
39 }
40
41 // 生产者类
42 class Productor implements Runnable {
43
44     private Clerk clerk;
45
46     public Productor(Clerk clerk) {
47         this.clerk = clerk;
48     }
49
50     @Override
51     public void run() {
52         for (int i = 0; i < 10; i++) {
53             clerk.get();
54         }
55     }
56 }
57
58 // 消费者类
59 class Consumer implements Runnable {
60
61     private Clerk clerk;
62
63     public Consumer(Clerk clerk) {
64         this.clerk = clerk;
65     }
66
67     @Override
68     public void run() {
69         for (int i = 0; i < 10; i++) {
70             clerk.sale();
71         }
72     }
73 }
74
75 public class TestProductorAndConsumer {
76
77     public static void main(String[] args) {
78         Clerk clerk = new Clerk();
79
80         Productor productor = new Productor(clerk);
81         Consumer consumer = new Consumer(clerk);
82
83         new Thread(productor, "Productor A").start();
84         new Thread(consumer, "Consumer B").start();
85     }
86 }

运行程序,不再死循环:

但是,如果现在有两个(多个)消费者线程和生产者线程,并且我们在生产者类的run方法中添加一个sleep()方法的执行,情况会如何呢?

 1 package concurrent;
 2
 3 //店员类
 4 class Clerk {
 5     private int product = 0;
 6
 7     // 进货
 8     public synchronized void get() {
 9         if (product >= 1) {
10             System.out.println("产品已满!");
11
12             // 等待
13             try {
14                 this.wait();
15             } catch (InterruptedException e) {
16                 e.printStackTrace();
17             }
18         }
19         System.out.println(Thread.currentThread().getName() + ":" + ++product);
20         // 唤醒
21         this.notifyAll();
22     }
23
24     // 售货
25     public synchronized void sale() {
26         if (product <= 0) {
27             System.out.println("缺货!");
28             // 等待
29             try {
30                 this.wait();
31             } catch (InterruptedException e) {
32                 e.printStackTrace();
33             }
34         }
35         System.out.println(Thread.currentThread().getName() + ":" + --product);
36         // 唤醒
37         this.notifyAll();
38     }
39 }
40
41 // 生产者类
42 class Productor implements Runnable {
43
44     private Clerk clerk;
45
46     public Productor(Clerk clerk) {
47         this.clerk = clerk;
48     }
49
50     @Override
51     public void run() {
52         for (int i = 0; i < 10; i++) {
53             try {
54                 Thread.sleep(100);
55             } catch (InterruptedException e) {
56                 // TODO Auto-generated catch block
57                 e.printStackTrace();
58             }
59             clerk.get();
60         }
61     }
62 }
63
64 // 消费者类
65 class Consumer implements Runnable {
66
67     private Clerk clerk;
68
69     public Consumer(Clerk clerk) {
70         this.clerk = clerk;
71     }
72
73     @Override
74     public void run() {
75         for (int i = 0; i < 10; i++) {
76             clerk.sale();
77         }
78     }
79 }
80
81 public class TestProductorAndConsumer {
82
83     public static void main(String[] args) {
84         Clerk clerk = new Clerk();
85
86         Productor productor = new Productor(clerk);
87         Consumer consumer = new Consumer(clerk);
88
89         new Thread(productor, "Productor A").start();
90         new Thread(consumer, "Consumer B").start();
91         new Thread(productor, "Productor C").start();
92         new Thread(consumer, "Consumer D").start();
93     }
94 }

运行程序:

产品数量出现了负数,这肯定是错误的。错误的原因在于,当一个消费者线程遇到产品为0时,等待,并释放锁标志,然后另外一个消费者线程获取到该锁标志,由于产品仍然为0,也等待,并释放锁标志。这时候,生产者线程获取到锁,在生产一个产品后,执行notifyAll()唤醒所有线程,这时候,一个消费者线程消费一个产品使得产品为0,另外一个消费者线程再消费一个产品使得产品变为了负数,这种现象称为虚假唤醒。在Object.wait()方法的javadoc中叙述了该如何解决这种问题:

即,将get和sale方法中的if都改为while,这样,每次被唤醒后,都会再次判断产品数是否>=0:

 1 package concurrent;
 2
 3 //店员类
 4 class Clerk {
 5     private int product = 0;
 6
 7     // 进货
 8     public synchronized void get() {
 9         while (product >= 1) {
10             System.out.println("产品已满!");
11
12             // 等待
13             try {
14                 this.wait();
15             } catch (InterruptedException e) {
16                 e.printStackTrace();
17             }
18         }
19         System.out.println(Thread.currentThread().getName() + ":" + ++product);
20         // 唤醒
21         this.notifyAll();
22     }
23
24     // 售货
25     public synchronized void sale() {
26         while (product <= 0) {
27             System.out.println("缺货!");
28             // 等待
29             try {
30                 this.wait();
31             } catch (InterruptedException e) {
32                 e.printStackTrace();
33             }
34         }
35         System.out.println(Thread.currentThread().getName() + ":" + --product);
36         // 唤醒
37         this.notifyAll();
38     }
39 }
40
41 // 生产者类
42 class Productor implements Runnable {
43
44     private Clerk clerk;
45
46     public Productor(Clerk clerk) {
47         this.clerk = clerk;
48     }
49
50     @Override
51     public void run() {
52         for (int i = 0; i < 10; i++) {
53             try {
54                 Thread.sleep(100);
55             } catch (InterruptedException e) {
56                 // TODO Auto-generated catch block
57                 e.printStackTrace();
58             }
59             clerk.get();
60         }
61     }
62 }
63
64 // 消费者类
65 class Consumer implements Runnable {
66
67     private Clerk clerk;
68
69     public Consumer(Clerk clerk) {
70         this.clerk = clerk;
71     }
72
73     @Override
74     public void run() {
75         for (int i = 0; i < 10; i++) {
76             clerk.sale();
77         }
78     }
79 }
80
81 public class TestProductorAndConsumer {
82
83     public static void main(String[] args) {
84         Clerk clerk = new Clerk();
85
86         Productor productor = new Productor(clerk);
87         Consumer consumer = new Consumer(clerk);
88
89         new Thread(productor, "Productor A").start();
90         new Thread(consumer, "Consumer B").start();
91         new Thread(productor, "Productor C").start();
92         new Thread(consumer, "Consumer D").start();
93     }
94 }

运行程序,发现结果终于正常了:

转载自:http://blog.csdn.net/xiangwanpeng/article/details/54973782

时间: 2024-10-02 00:01:22

【转】通过生产者消费者案例理解等待唤醒机制和虚假唤醒的相关文章

Condition线程通信_生产者消费者案例

①Condition 接口描述了可能会与锁有关联的条件变量. 这些变量在用 法上与使用 Object.wait 访问的隐式监视器类似,但提供了更强大的 功能. 需要特别指出的是,单个 Lock 可能与多个 Condition 对象关 联. 为了避免兼容性问题,Condition 方法的名称与对应的 Object 版 本中的不同.② 在 Condition 对象中,与 wait.notify 和 notifyAll 方法对应的分别是 await.signal 和 signalAll.③ Condi

java多线程 生产者消费者案例-虚假唤醒

package com.java.juc; public class TestProductAndConsumer { public static void main(String[] args) { Clerk clerk = new Clerk(); Produce produce = new Produce(clerk); Consumer consumer = new Consumer(clerk); new Thread(produce, "线程A").start(); ne

多线程生产者消费者案例

题目需求:写一个生产者消费者容器,支持多个生产者消费者同时访问,容器里最多放十个数,需要get()和put()方法,当容器中没数据时,生产者开始生产数据,消费者等待,数据量满十个时,生产者等待,消费者开始消费. /** * Created by canner on 2018/11/30. */ public class MyContainer<T> { private final int MAX = 10; private final LinkedList<T> list = ne

Java多线程虚假唤醒问题(生产者和消费者关系)

何为虚假唤醒: 当一个条件满足时,很多线程都被唤醒了,但是只有其中部分是有用的唤醒,其它的唤醒都是无用功:比如买货:如果商品本来没有货物,突然进了一件商品,这是所有的线程都被唤醒了,但是只能一个人买,所以其他人都是假唤醒,获取不到对象的锁: 避免虚假唤醒: 避免虚假唤醒的示例:这里使用了 Lambda 表达式 package com.jia.pc; public class A { public static void main(String[] args) { Data data = new

(三)juc高级特性——虚假唤醒 / Condition / 按序交替 / ReadWriteLock / 线程八锁

8. 生产者消费者案例-虚假唤醒 参考下面生产者消费者案例: /* * 生产者和消费者案例 */ public class TestProductorAndConsumer { public static void main(String[] args) { Clerk clerk = new Clerk(); Productor pro = new Productor(clerk); Consumer cus = new Consumer(clerk); new Thread(pro, "生产

java基础知识回顾之java Thread类学习(八)--java多线程通信等待唤醒机制经典应用(生产者消费者)

 *java多线程--等待唤醒机制:经典的体现"生产者和消费者模型 *对于此模型,应该明确以下几点: *1.生产者仅仅在仓库未满的时候生产,仓库满了则停止生产. *2.消费者仅仅在有产品的时候才能消费,仓空则等待. *3.当消费者发现仓储没有产品可消费的时候,会唤醒等待生产者生产. *4.生产者在生产出可以消费的产品的时候,应该通知等待的消费者去消费. 下面先介绍个简单的生产者消费者例子:本例只适用于两个线程,一个线程生产,一个线程负责消费. 生产一个资源,就得消费一个资源. 代码如下: pub

java多线程中的生产者与消费者之等待唤醒机制@Version1.0

一.生产者消费者模式的学生类成员变量生产与消费demo,第一版1.等待唤醒:    Object类中提供了三个方法:    wait():等待    notify():唤醒单个线程    notifyAll():唤醒所有线程2.为什么这些方法不定义在Thread类中呢?  这些方法的调用必须通过锁对象调用,而我们刚才使用的锁对象是任意锁对象.  所以,这些方法必须定义在Object类中.3.当我们在使用多线程的时候有的时候需要,一条线程产生一个数据,另一条线程接着消费一个数据,一边生产一边消费,

生产者与消费者案例-虚假唤醒

以下是一个案例,有一个店员,负责进货和卖货.进货生产,卖货消费. 当商品超过10件,生产等待,消费继续,当少于0件,消费等待,消费继续. 正常代码如下: package com.atguigu.juc; /* * 生产者和消费者案例 */ public class TestProductorAndConsumer { public static void main(String[] args) { Clerk clerk = new Clerk(); Productor pro = new Pr

Android-Java多线程通讯(生产者 消费者)&amp;10条线程对-等待唤醒/机制的管理

上一篇博客 Android-Java多线程通讯(生产者 消费者)&等待唤醒机制 是两条线程(Thread-0 / Thread-1) 在被CPU随机切换执行: 而今天这篇博客是,在上一篇博客Android-Java多线程通讯(生产者 消费者)&等待唤醒机制 的基础上,扩大规模增加10条线程去执行 生产者 消费者: 注意:?? 上一篇博客是两条线程在执行(生产者 消费者)例如:当Thread-0 锁.wait(); 等待 冻结后,  Thread-1 锁.notify(); 唤醒的一定是 T