生产者和消费者模型是操作系统中经典的同步问题。该问题最早由Dijkstra提出,用以演示它提出的信号量机制。

经典的生产者和消费者模型的描写叙述是:有一群生产者进程在生产产品。并将这些产品提供给消费者进程去消费。为使生产者进程与消费者进程能并发执行,在两者之间设置了一个具有n个缓冲区的缓冲池,生产者进程将它所生产的产品放入一个缓冲区中。消费者进程可从一个缓冲区中取走产品去消费。虽然全部的生产者进程和消费者进程都是以异步方式执行的。但它们之间必须保持同步,即不同意消费者进程到一个空缓冲区去取产品,也不同意生产者进程向一个已装满产品且尚未被取走的缓冲区投放产品。

首先我们复习一下操作系统中同步机制中应遵循的准则:

  1. 空暇让进:当无进程处于临界区时,应同意一个请求进入临界区的进程进入临界区;
  2. 忙则等待:当已有进程进入临界区时,其它试图进入临界区的进程必须等待。
  3. 有限等待:对要求訪问临界资源的进程,应保证在有限时间内能进入自己的临界区。以免陷入“死等”状态。
  4. 让权等待:当进程不能进入自己的临界区时,应马上释放处理机。以免进程陷入“忙等”;

    在生产者和消费者模型中要保证一下几点:

    1.生产者在往缓存队列中放产品时,消费者不能取产品。

    2.消费者从缓存队列中取产品时。生产者不能放产品。

    3.同一时刻仅仅有一个生产者能够往缓存队列中放产品。

    4.同一时刻仅仅有一个消费者能够从缓存队列中取产品。

    5.缓存队列满时生产者不能往缓存队列中放产品。

    6.缓存队列为空时消费者不能从缓存队列中取产品。

本样例中的缓存队列模仿java jdk中的ArrayBlockingQueue,这是一个堵塞队列,缓存池满时会自己主动将生产者线程挂起,缓存池空时会自己主动将消费者线程挂起。

缓存池

public class Pool<E> {

    /**队列最长长度*/
private int MaxSize = 1000; /**队列默认长度*/
private static final int defaultSize = 100; /**资源池*/
private Object[] objs ; /**队头*/
private int front; /**队尾*/
private int rear; /**元素的个数*/
private int nItems; /** Main lock guarding all access */
final ReentrantLock lock; /** Condition for waiting takes */
private final Condition notEmpty; /** Condition for waiting puts */
private final Condition notFull; private int useSize = 0; public Pool() {
this(defaultSize);
useSize = defaultSize;
} public Pool(int size) {
if(size < 0)
throw new IndexOutOfBoundsException();
size = size > MaxSize ? MaxSize : size;
useSize = size;
objs = new Object[size];
front = 0;
rear = -1;
nItems = 0; lock = new ReentrantLock(true);
notEmpty = lock.newCondition();
notFull = lock.newCondition(); } /**进队*/
private void queue(E e) {
if(rear == useSize - 1)
rear = -1;
objs[++rear] = e;
nItems++;
notEmpty.signal();
} /**出队*/
private E dequeue() {
E e = (E)objs[front++];
if(front == useSize)
front = 0;
nItems--;
notFull.signal();
return e;
} /**进队 资源池满会将入队线程挂起*/
public void offer(E e) throws InterruptedException {
final ReentrantLock lock = this.lock;
lock.lock();
try {
while(nItems == objs.length)
notFull.await();
queue(e);
System.out.println("学生进队。当前池中有 " + nItems + " 名同学" );
} finally {
lock.unlock();
} } /**出队 资源池空会将出队线程挂起*/
public E poll() throws InterruptedException {
final ReentrantLock lock = this.lock;
lock.lock();
try {
while(nItems == 0)
notEmpty.await();
E e = dequeue();
System.out.println("学生出队,当前池中有 " + nItems + " 名同学" );
return e;
} finally {
lock.unlock();
}
} /**是否满*/
public boolean isFull() {
final ReentrantLock lock = this.lock;
lock.lock();
try {
return nItems == MaxSize ? true : false;
} finally {
lock.unlock();
} } /**推断是否为空*/
public boolean isEmpty() {
final ReentrantLock lock = this.lock;
lock.lock();
try {
return nItems == 0 ? true : false;
} finally {
lock.unlock();
} } /**返回队列中元素个数*/
public int size() {
final ReentrantLock lock = this.lock;
lock.lock();
try {
return this.nItems;
} finally {
lock.unlock();
} }
}

測试模型

public class Student {

    private String name;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
} }

主类

public class PM {

    private Pool<Student> pools = new Pool<Student>(1000);

    public static void main(String[] args) {
PM pm = new PM();
ExecutorService executor = Executors.newFixedThreadPool(6);
executor.execute(pm.new consume());
executor.execute(pm.new consume());
executor.execute(pm.new consume());
executor.execute(pm.new produce());
executor.execute(pm.new produce());
executor.execute(pm.new produce());
} class produce implements Runnable { @Override
public void run() {
while(true) {
try {
pools.offer(new Student());
} catch (InterruptedException e) {
e.printStackTrace();
}
} } } class consume implements Runnable { @Override
public void run() {
while(true) {
try {
pools.poll();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
} } }

执行结果:

最新文章

  1. Light OJ 1029- Civil and Evil Engineer (图论-最小生成树)
  2. bzoj2424 [HAOI2010]订货
  3. quick cocos2dx lua 内存释放
  4. jQuery 常用的代码片段
  5. 编程内功修炼之数据结构—BTree(三)总结
  6. Androidclient与服务端(jsp)之间json的传输与解析【附效果图附源代码】
  7. nyoj 517 最小公倍数 【java睑板】
  8. iwebshop里面传数组且输出
  9. [UWP]分享一个基于HSV色轮的调色板应用
  10. ElasticSearch 学习记录之ES短语匹配基本用法
  11. 文件解压缩 tar zip
  12. Python正则表达式操作指南(转)
  13. hdu 3360 最小点覆盖 **
  14. Asp.net web Control Enable 属性设置
  15. SharpGL学习笔记(十) 常见的光源类型,创建光源
  16. IntelliJ IDEA 去除IDE自动的参数名 提示功能
  17. php取得当前时间函数
  18. i.mx6 Android5.1.1 初始化流程之框架
  19. Java-贪心算法
  20. js中定义变量的三种方式const,val,let 的区别

热门文章

  1. GDSOI2019划水记
  2. 使用maven的tomcat:run进行web项目热部署
  3. java JDK设置环境变量
  4. Lightoj 1127 - Funny Knapsack 【二分】
  5. ANSI-X99MAC算法和PBOC的3DES MAC算法
  6. Codeforces 441 B. Valera and Fruits
  7. 通过OpenSSL解析X509证书基本项
  8. 安卓View的缓冲机制
  9. ubuntu-date命令的使用
  10. FragMent-通过Arguments方法 跟activity通信