Java - 生产者消费者问题
概要
本章,会对“生产/消费者问题”进行讨论。涉及到的内容包括:
1. 生产/消费者模型
2. 生产/消费者实现
转载请注明出处:http://www.cnblogs.com/skywang12345/p/3480016.html
1. 生产/消费者模型
生产/消费者问题是个非常典型的多线程问题,涉及到的对象包括“生产者”、“消费者”、“仓库”和“产品”。他们之间的关系如下:
(01) 生产者仅仅在仓储未满时候生产,仓满则停止生产。
(02) 消费者仅仅在仓储有产品时候才能消费,仓空则等待。
(03) 当消费者发现仓储没产品可消费时候会通知生产者生产。
(04) 生产者在生产出可消费产品时候,应该通知等待的消费者去消费。
2. 生产/消费者实现
下面通过wait()/notify()方式实现该模型(后面在学习了线程池相关内容之后,再通过其它方式实现生产/消费者模型)。源码如下:

1 // Demo1.java
2 // 仓库
3 class Depot {
4 private int capacity; // 仓库的容量
5 private int size; // 仓库的实际数量
6
7 public Depot(int capacity) {
8 this.capacity = capacity;
9 this.size = 0;
10 }
11
12 public synchronized void produce(int val) {
13 try {
14 // left 表示“想要生产的数量”(有可能生产量太多,需多此生产)
15 int left = val;
16 while (left > 0) {
17 // 库存已满时,等待“消费者”消费产品。
18 while (size >= capacity)
19 wait();
20 // 获取“实际生产的数量”(即库存中新增的数量)
21 // 如果“库存”+“想要生产的数量”>“总的容量”,则“实际增量”=“总的容量”-“当前容量”。(此时填满仓库)
22 // 否则“实际增量”=“想要生产的数量”
23 int inc = (size+left)>capacity ? (capacity-size) : left;
24 size += inc;
25 left -= inc;
26 System.out.printf("%s produce(%3d) --> left=%3d, inc=%3d, size=%3d\n",
27 Thread.currentThread().getName(), val, left, inc, size);
28 // 通知“消费者”可以消费了。
29 notifyAll();
30 }
31 } catch (InterruptedException e) {
32 }
33 }
34
35 public synchronized void consume(int val) {
36 try {
37 // left 表示“客户要消费数量”(有可能消费量太大,库存不够,需多此消费)
38 int left = val;
39 while (left > 0) {
40 // 库存为0时,等待“生产者”生产产品。
41 while (size <= 0)
42 wait();
43 // 获取“实际消费的数量”(即库存中实际减少的数量)
44 // 如果“库存”<“客户要消费的数量”,则“实际消费量”=“库存”;
45 // 否则,“实际消费量”=“客户要消费的数量”。
46 int dec = (size<left) ? size : left;
47 size -= dec;
48 left -= dec;
49 System.out.printf("%s consume(%3d) <-- left=%3d, dec=%3d, size=%3d\n",
50 Thread.currentThread().getName(), val, left, dec, size);
51 notifyAll();
52 }
53 } catch (InterruptedException e) {
54 }
55 }
56
57 public String toString() {
58 return "capacity:"+capacity+", actual size:"+size;
59 }
60 }
61
62 // 生产者
63 class Producer {
64 private Depot depot;
65
66 public Producer(Depot depot) {
67 this.depot = depot;
68 }
69
70 // 消费产品:新建一个线程向仓库中生产产品。
71 public void produce(final int val) {
72 new Thread() {
73 public void run() {
74 depot.produce(val);
75 }
76 }.start();
77 }
78 }
79
80 // 消费者
81 class Customer {
82 private Depot depot;
83
84 public Customer(Depot depot) {
85 this.depot = depot;
86 }
87
88 // 消费产品:新建一个线程从仓库中消费产品。
89 public void consume(final int val) {
90 new Thread() {
91 public void run() {
92 depot.consume(val);
93 }
94 }.start();
95 }
96 }
97
98 public class Demo1 {
99 public static void main(String[] args) {
100 Depot mDepot = new Depot(100);
101 Producer mPro = new Producer(mDepot);
102 Customer mCus = new Customer(mDepot);
103
104 mPro.produce(60);
105 mPro.produce(120);
106 mCus.consume(90);
107 mCus.consume(150);
108 mPro.produce(110);
109 }
110 }

说明:
(01) Producer是“生产者”类,它与“仓库(depot)”关联。当调用“生产者”的produce()方法时,它会新建一个线程并向“仓库”中生产产品。
(02) Customer是“消费者”类,它与“仓库(depot)”关联。当调用“消费者”的consume()方法时,它会新建一个线程并消费“仓库”中的产品。
(03) Depot是“仓库”类,仓库中记录“仓库的容量(capacity)”以及“仓库中当前产品数目(size)”。
“仓库”类的生产方法produce()和消费方法consume()方法都是synchronized方法,进入synchronized方法体,意味着这个线程获取到了该“仓库”对象的同步锁。这也就是说,同一时间,生产者和消费者线程只能有一个能运行。通过同步锁,实现了对“残酷”的互斥访问。
对于生产方法produce()而言:当仓库满时,生产者线程等待,需要等待消费者消费产品之后,生产线程才能生产;生产者线程生产完产品之后,会通过notifyAll()唤醒同步锁上的所有线程,包括“消费者线程”,即我们所说的“通知消费者进行消费”。
对于消费方法consume()而言:当仓库为空时,消费者线程等待,需要等待生产者生产产品之后,消费者线程才能消费;消费者线程消费完产品之后,会通过notifyAll()唤醒同步锁上的所有线程,包括“生产者线程”,即我们所说的“通知生产者进行生产”。
(某一次)运行结果:

Thread-0 produce( 60) --> left= 0, inc= 60, size= 60
Thread-4 produce(110) --> left= 70, inc= 40, size=100
Thread-2 consume( 90) <-- left= 0, dec= 90, size= 10
Thread-3 consume(150) <-- left=140, dec= 10, size= 0
Thread-1 produce(120) --> left= 20, inc=100, size=100
Thread-3 consume(150) <-- left= 40, dec=100, size= 0
Thread-4 produce(110) --> left= 0, inc= 70, size= 70
Thread-3 consume(150) <-- left= 0, dec= 40, size= 30
Thread-1 produce(120) --> left= 0, inc= 20, size= 50

Java - 生产者消费者问题的更多相关文章
- 基于Java 生产者消费者模式(详细分析)
Java 生产者消费者模式详细分析 本文目录:1.等待.唤醒机制的原理2.Lock和Condition3.单生产者单消费者模式4.使用Lock和Condition实现单生产单消费模式5.多生产多消费模 ...
- Java生产者消费者的三种实现
Java生产者消费者是最基础的线程同步问题,java岗面试中还是很容易遇到的,之前没写过多线程的代码,面试中被问到很尬啊,面完回来恶补下.在网上查到大概有5种生产者消费者的写法,分别如下. 用sync ...
- java 生产者消费者问题 并发问题的解决
引言 生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况: 生产者消费者图 ...
- Java生产者消费者模型
在Java中线程同步的经典案例,不同线程对同一个对象同时进行多线程操作,为了保持线程安全,数据结果要是我们期望的结果. 生产者-消费者模型可以很好的解释这个现象:对于公共数据data,初始值为0,多个 ...
- java 生产者消费者问题 并发问题的解决(转)
引言 生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况: 生产者消费者图 ...
- Java 生产者消费者模式详细分析
*/ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...
- Java生产者消费者问题
1. package interview.thread; import java.util.LinkedList; import java.util.Queue; import org.apache. ...
- Java生产者消费者模式
为什么要使用生产者和消费者模式 在线程世界里,生产者就是生产数据的线程,消费者就是消费数据的线程.在多线程开发当中,如果生产者处理速度很快,而消费者处理速度很慢,那么生产者就必须等待消费者处理完,才能 ...
- JAVA生产者消费者的实现
春节回了趟老家,又体验了一次流水席,由于桌席多,导致上菜慢,于是在等待间,总结了一下出菜流程的几个特点: 1.有多个灶台,多个灶台都在同时做菜出来. 2.做出来的菜,会有专人用一个托盘端出来,每次端出 ...
- java生产者消费者并发协作
随着职务转变,代码荒废很久了,很多时间都是在沟通需求,作为一名技术员,不写代码就感觉是在自废武功,慢慢颓废了很多,今天重新回顾了下JAVA线程知识,基础知识就不梳理了,网上也很多,主要关键几个状态位( ...
随机推荐
- JQuery的页面操作
window.location = "http://www.xxxxxxxx.net" 跳转后有后退功能 其实应该是 window.location.hrefwindow.loca ...
- linux服务器ssh免密码登录
假设有三台服务器的ip分别是10.9.1.101.10.9.1.102.10.9.1.103 修改hosts文件 分别修改三台服务器的hosts的文件(vi /etc/hosts), 在hosts文件 ...
- Sublime Text3 实现在浏览器中以HTML格式预览md文件
1.首先找到Package Control 打开Sublime Text3,找到菜单栏:Preferences → Package Control,没有找到Package Control,那么点击Pa ...
- MATLAB入门(一)数组
特殊变量: 数组的创建: %% 数组的生成(带:不在命令行显示结果) x= rand() ; %随机生成3*3的矩阵,矩阵数的值在[,] x(:,::); %取所有行,(从第1列开始:步长为2:到第 ...
- Spark程序提交到Yarn集群时所遇异常
Exception 1:当我们将任务提交给Spark Yarn集群时,大多会出现以下异常,如下: 14/08/09 11:45:32 WARN component.AbstractLifeCycle: ...
- JVM中强引用,弱引用,软引用和幽灵引用的代码
上代码: public class ReferenceTest { public static void main(String[] args) { //test1();//软引用 //test2() ...
- android开发学习——day4
自己手动创建空活动,创建和加载布局,效果:界面中出现靠上对齐的button 在活动中使用Toast,效果:对点击按钮做出响应 在活动中使用menu,效果:界面中出现菜单,并且点击对应选项会有响应 De ...
- 搭建Golang开发环境
Go语言是谷歌 2009 年首次推出并在 2012 年正式发布的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性.谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发 ...
- Python学习--14 序列化
把变量从内存中变成可存储或传输的过程称之为序列化,在Python中叫pickling,在其他语言中也被称之为serialization,marshalling,flattening等等. pickle ...
- CentOS安装PHP7+Nginx+MySQL
本文属于动手搭建PHP开发环境的一部分,更多点击链接查看. 本文以centos6为例.命令部分均省略sudo命令. 安装PHP 下载 http://cn2.php.net/distributions/ ...