生产消费模型(2)await() / signal()方法

本文参考于湘下农人的文章《Java并发:await/wait与锁》
在JDK5中,用ReentrantLock和Condition可以实现等待/通知模型,具有更大的灵活性。通过在Lock对象上调用newCondition()方法,将条件变量和一个锁对象进行绑定,进而控制并发程序访问竞争资源的安全。

await/waitSleepYield
是否释放持有的锁释放不释放不释放
调用后何时恢复唤醒后进入就绪态指定时间后立刻进入就绪态
谁的方法Condition/ObjectThreadThread
执行环境同步代码块任意位置任意位置

在第一种方法中,我是使用Object 的wait和notify实现同步的。举例来说,在生产消费模型中,consumer发现没有东西了,等待,produer生成东西后,唤醒。
在这里插入图片描述
有了lock后,世道变了,现在是:
在这里插入图片描述
区别主要有3点:

  1. 不再用synchronize把同步代码包装起来,而是用lock对象调用lock()方法;
  2. 阻塞需要条件对象condition;
  3. 等待和唤醒的对象是condition而不是lock,对应的方法是await和signal,而不是wait和notify。

为什么需要使用condition呢?简单一句话,lock更灵活。以前的方式只能有一个等待队列,在实际应用时可能需要多个,比如读和写。为了这个灵活性,lock将同步互斥控制和等待队列分离开来,互斥保证在某个时刻只有一个线程访问临界区(lock自己完成),等待队列负责保存被阻塞的线程(condition完成)。

通过查看ReentrantLock的源代码发现,condition其实是等待队列的一个管理者,condition确保阻塞的对象按顺序被唤醒。

代码实现:只改动仓库类

package 生产消费模型;

import java.util.LinkedList;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class 仓库 {
	private final int MAX_SIZE = 20;//仓库最大容量
	private LinkedList<Object> list = new LinkedList<>();//仓库存储介质
	private final Lock lock = new ReentrantLock();  //锁
	private final Condition full = lock.newCondition();  //仓库满的条件变量
	private final Condition empty = lock.newCondition();  //仓库空的条件变量
	
	public void produce(){
			lock.lock();  //不再是synchronized (list) {}结构的同步代码块
			while(list.size()==MAX_SIZE){
				System.out.println("【生产者"+Thread.currentThread().getName()+"】仓库已满");
				try {
					full.await();//若满,full条件变量等待
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
			list.add(new Object());
			System.out.println("【生产者"+Thread.currentThread().getName()+"】生产一个产品,现库存"+list.size());
			empty.signalAll();//空条件变量唤醒
			lock.unlock();//记得解锁
	}
	public void consume(){
			lock.lock();
			while(list.size()==0){
				System.out.println("【消费者"+Thread.currentThread().getName()+"】仓库为空");
				try {
					empty.wait();
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				};
			}
			list.remove();
			System.out.println("【消费者" + Thread.currentThread().getName()+ "】消费一个产品,现库存" + list.size());
			full.signalAll();
			lock.unlock();
	}
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值