生产者与消费者(二)---await与 signal
前面阐述了实现生产者与消费者问题的一种方式:wait() / notify()方法,本文继续阐述多线程的经典问题---生产者与消费者的第二种方式:await() / signal()方法。
await() / signal()方法
在JDK5.0之后,Java提供了更加健壮的线程处理机制,包括同步、锁定、线程池等,它们可以实现更细粒度的线程控制。await()和signal()就是其中用来做同步的两种方法,它们的功能基本上和wait() / nofity()相同,完全可以取代它们,但是它们和新引入的锁定机制Lock直接挂钩,具有更大的灵活性。通过在Lock对象上调用newCondition()方法,将条件变量和一个锁对象进行绑定,进而控制并发程序访问竞争资源的安全。
缓冲区(仓库):
import java.util.LinkedList;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class Storage {
// 仓库最大存储量
private final int MAX_SIZE = 100; // 仓库存储的载体
private LinkedList<Object> list = new LinkedList<Object>(); // 锁
private final Lock lock = new ReentrantLock(); // 仓库满的条件变量
private final Condition full = lock.newCondition(); // 仓库空的条件变量
private final Condition empty = lock.newCondition(); // 生产num个产品
public void produce(int num){
// 获得锁
lock.lock(); // 如果仓库剩余容量不足
while (list.size() + num > MAX_SIZE){
System.out.println("【要生产的产品数量】:" + num + "【库存量】:" + list.size()+ "暂时不能执行生产任务!");
try{
// 由于条件不满足,生产阻塞
full.await();
}
catch (InterruptedException e){
e.printStackTrace();
}
} // 生产条件满足情况下,生产num个产品
for (int i = 1; i <= num; ++i){
list.add(new Object());
} System.out.println("【已经生产产品数】:" + num + "/t【现仓储量为】:" + list.size()); // 唤醒其他所有线程
full.signalAll();
empty.signalAll(); // 释放锁
lock.unlock();
} // 消费num个产品
public void consume(int num)
{
// 获得锁
lock.lock(); // 如果仓库存储量不足
while (list.size() < num){
System.out.println("【要消费的产品数量】:" + num + "【库存量】:" + list.size()+ "暂时不能执行生产任务!");
try{
// 由于条件不满足,消费阻塞
empty.await();
}
catch (InterruptedException e){
e.printStackTrace();
}
} // 消费条件满足情况下,消费num个产品
for (int i = 1; i <= num; ++i){
list.remove();
} System.out.println("【已经消费产品数】:" + num + "【现仓储量为】:" + list.size()); // 唤醒其他所有线程
full.signalAll();
empty.signalAll(); // 释放锁
lock.unlock();
} // set/get方法
public int getMAX_SIZE()
{
return MAX_SIZE;
} public LinkedList<Object> getList()
{
return list;
} public void setList(LinkedList<Object> list)
{
this.list = list;
}
}
生产者:
public class Producer extends Thread {
// 每次生产的产品数量
private int num; // 所在放置的仓库
private Storage storage; // 构造函数,设置仓库
public Producer(Storage storage){
this.storage = storage;
} // 线程run函数
public void run(){
produce(num);
} // 调用仓库Storage的生产函数
public void produce(int num){
storage.produce(num);
} // get/set方法
public int getNum()
{
return num;
} public void setNum(int num)
{
this.num = num;
} public Storage getStorage()
{
return storage;
} public void setStorage(Storage storage)
{
this.storage = storage;
} }
消费者:
public class Consumer extends Thread {
// 每次消费的产品数量
private int num; // 所在放置的仓库
private Storage storage; // 构造函数,设置仓库
public Consumer(Storage storage){
this.storage = storage;
} // 线程run函数
public void run(){
consume(num);
} // 调用仓库Storage的生产函数
public void consume(int num){
storage.consume(num);
} // get/set方法
public int getNum()
{
return num;
} public void setNum(int num)
{
this.num = num;
} public Storage getStorage()
{
return storage;
} public void setStorage(Storage storage)
{
this.storage = storage;
}
}
测试类:
public class Test {
public static void main(String[] args) {
// 仓库对象
Storage storage = new Storage(); // 生产者对象
Producer p1 = new Producer(storage);
Producer p2 = new Producer(storage);
Producer p3 = new Producer(storage);
Producer p4 = new Producer(storage);
Producer p5 = new Producer(storage);
Producer p6 = new Producer(storage);
Producer p7 = new Producer(storage); // 消费者对象
Consumer c1 = new Consumer(storage);
Consumer c2 = new Consumer(storage);
Consumer c3 = new Consumer(storage); // 设置生产者产品生产数量
p1.setNum(10);
p2.setNum(10);
p3.setNum(10);
p4.setNum(10);
p5.setNum(10);
p6.setNum(10);
p7.setNum(80); // 设置消费者产品消费数量
c1.setNum(50);
c2.setNum(20);
c3.setNum(30); // 线程开始执行
c1.start();
c2.start();
c3.start();
p1.start();
p2.start();
p3.start();
p4.start();
p5.start();
p6.start();
p7.start();
}
}
生产者与消费者(二)---await与 signal的更多相关文章
- 母鸡下蛋实例:多线程通信生产者和消费者wait/notify和condition/await/signal条件队列
简介 多线程通信一直是高频面试考点,有些面试官可能要求现场手写生产者/消费者代码来考察多线程的功底,今天我们以实际生活中母鸡下蛋案例用代码剖析下实现过程.母鸡在鸡窝下蛋了,叫练从鸡窝里把鸡蛋拿出来这个 ...
- JAVA基础再回首(二十五)——Lock锁的使用、死锁问题、多线程生产者和消费者、线程池、匿名内部类使用多线程、定时器、面试题
JAVA基础再回首(二十五)--Lock锁的使用.死锁问题.多线程生产者和消费者.线程池.匿名内部类使用多线程.定时器.面试题 版权声明:转载必须注明本文转自程序猿杜鹏程的博客:http://blog ...
- 利用反射快速给Model实体赋值 使用 Task 简化异步编程 Guid ToString 格式知多少?(GUID 格式) Parallel Programming-实现并行操作的流水线(生产者、消费者) c# 无损高质量压缩图片代码 8种主要排序算法的C#实现 (一) 8种主要排序算法的C#实现 (二)
试想这样一个业务需求:有一张合同表,由于合同涉及内容比较多所以此表比较庞大,大概有120多个字段.现在合同每一次变更时都需要对合同原始信息进行归档一次,版本号依次递增.那么我们就要新建一张合同历史表, ...
- Java 多线程基础(十二)生产者与消费者
Java 多线程基础(十二)生产者与消费者 一.生产者与消费者模型 生产者与消费者问题是个非常典型的多线程问题,涉及到的对象包括“生产者”.“消费者”.“仓库”和“产品”.他们之间的关系如下: ①.生 ...
- Java并发编程(4)--生产者与消费者模式介绍
一.前言 这种模式在生活是最常见的,那么它的场景是什么样的呢? 下面是我假象的,假设有一个仓库,仓库有一个生产者和一个消费者,消费者过来消费的时候会检测仓库中是否有库存,如果没有了则等待生产,如果有就 ...
- JUC 并发编程--02,生产者和消费者 synchronized的写法 , juc的写法. Condition的用法
synchronized的写法 class PCdemo{ public static void main(String[] args) { //多个线程操作同一资源 Data data = new ...
- Java实现生产者和消费者
生产者和消费者问题是操作系统的经典问题,在实际工作中也常会用到,主要的难点在于协调生产者和消费者,因为生产者的个数和消费者的个数不确定,而生产者的生成速度与消费者的消费速度也不一样,同时还要实现生产者 ...
- java中的生产者和消费者的问题
1----使用Java.util.concurrent.locks包中的lock接口取代synchronized,通过ReentrantLock这个已经实现Lock接口的类, 创建ReentrantL ...
- 生产者与消费者(三)---BlockingQueue
前面阐述了实现生产者与消费者问题的两种方式:wait() / notify()方法 和 await() / signal()方法,本文继续阐述多线程的经典问题---生产者与消费者的第三种方式:Bloc ...
随机推荐
- Java中url传递中文参数取值乱码的解决方法
java中URL参数中有中文值,传到服务端,在用request.getParameter()方法,得到的常常会是乱码,这将涉及到字符解码操作. 方法一: http://xxx.do?ptname=’我 ...
- man命令
man,这个命令,非常好!后续,更新
- __attribute__机制介绍 (转)
1. __attribute__ GNU C的一大特色(却不被初学者所知)就是__attribute__机制. __attribute__可以设置函数属性(Function Attribute).变量 ...
- hdoj 1564 Play a game
Play a game Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- 几款开源的图形化Redis客户端管理软件推荐
Redis是一个超精简的基于内存的键值对数据库(key-value),一般对并发有一定要求的应用都用其储存session,乃至整个数据库.不过它公自带一个最小化的命令行式的数据库管理工具,有时侯使用起 ...
- Mina学习之IoSession
Session(会话)是Mina的核心部分:每当一个clinent连接到server时,都会创建一个新的session,并且保存在内存中知道该链接断开. session 是用来存储一些关于连接信息,加 ...
- WPF button 圆角制作
将以下节点复制到app.xaml的<Application.Resources>节点下 <Style TargetType="{x:Type Button}"&g ...
- Java-struts2的问题 java.lang.NoClassDefFoundError: org/apache/commons/lang3/StringUtils
缺commons-lang3-3.1.jar,添加之后就可以了
- End-to-End Tracing of Ajax/Java Applications Using DTrace
End-to-End Tracing of Ajax/Java Applications Using DTrace By Amit Hurvitz, July 2007 Aja ...
- 精《记叙“tom”4年的软件开发之旅》
1.引言 本篇文章是记叙tom四年的软件开发从业经历,虽然他资历不长,况且本身也是个菜鸟,但他也曾有过荣誉.迷茫.困惑与选择,在这里他希望通过自己所经历过的事情分享给大家,给那些真正热爱软件开发的同学 ...