作者要的是一个生产者生成,接着必须有一个消费者消费,那这不是需要单线程吗?或者使用1个大小的阻塞队列。所以只谈论问题本身,不谈论好不好。

具体代码:

  1. import java.util.concurrent.locks.Condition;
  2. import java.util.concurrent.locks.Lock;
  3. import java.util.concurrent.locks.ReentrantLock;
  4. //生产/消费者模式
  5. public class Basket {
  6. Lock lock = new ReentrantLock();
  7. // 产生Condition对象
  8. Condition produced = lock.newCondition();
  9. Condition consumed = lock.newCondition();
  10. boolean available = false;
  11. public void produce() throws InterruptedException {
  12. lock.lock();
  13. try {
  14. if (available) {
  15. produced.await(); // 放弃lock进入睡眠
  16. }
  17. System.out.println("Apple produced.");
  18. available = true;
  19. consumed.signal(); // 发信号唤醒等待这个Condition的线程
  20. } finally {
  21. lock.unlock();
  22. }
  23. }
  24. public void consume() throws InterruptedException {
  25. lock.lock();
  26. try {
  27. if (!available) {
  28. consumed.await(); // 放弃lock进入睡眠
  29. }
  30. /* 吃苹果 */
  31. System.out.println("Apple consumed.");
  32. available = false;
  33. produced.signal(); // 发信号唤醒等待这个Condition的线程
  34. } finally {
  35. lock.unlock();
  36. }
  37. }
  38. }
import java.util.concurrent.locks.Condition;

import java.util.concurrent.locks.Lock;

import java.util.concurrent.locks.ReentrantLock;

//生产/消费者模式

public class Basket {

    Lock lock = new ReentrantLock();

    // 产生Condition对象

    Condition produced = lock.newCondition();

    Condition consumed = lock.newCondition();

    boolean available = false;

    public void produce() throws InterruptedException {

        lock.lock();

        try {

            if (available) {

                produced.await(); // 放弃lock进入睡眠

            }

            System.out.println("Apple produced.");

            available = true;

            consumed.signal(); // 发信号唤醒等待这个Condition的线程

        } finally {

            lock.unlock();

        }

    }

    public void consume() throws InterruptedException {

        lock.lock();

        try {

            if (!available) {

                consumed.await(); // 放弃lock进入睡眠

            }

            /* 吃苹果 */

            System.out.println("Apple consumed.");

            available = false;

            produced.signal(); // 发信号唤醒等待这个Condition的线程

        } finally {

            lock.unlock();

        }

    }

}
  1. import java.util.concurrent.ExecutorService;
  2. import java.util.concurrent.Executors;
  3. //测试用类
  4. public class ConditionTester {
  5. public static void main(String[] args) throws InterruptedException {
  6. final Basket basket = new Basket();
  7. // 定义一个producer
  8. Runnable producer = new Runnable() {
  9. public void run() {
  10. try {
  11. basket.produce();
  12. } catch (InterruptedException ex) {
  13. ex.printStackTrace();
  14. }
  15. }
  16. };
  17. // 定义一个consumer
  18. Runnable consumer = new Runnable() {
  19. public void run() {
  20. try {
  21. basket.consume();
  22. } catch (InterruptedException ex) {
  23. ex.printStackTrace();
  24. }
  25. }
  26. };
  27. // 各产生10个consumer和producer
  28. ExecutorService service = Executors.newCachedThreadPool();
  29. for (int i = ; i < ; i++)
  30. service.submit(consumer);
  31. Thread.sleep( * );
  32. for (int i = ; i < ; i++)
  33. service.submit(producer);
  34. service.shutdown();
  35. }
  36. }
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; //测试用类
public class ConditionTester { public static void main(String[] args) throws InterruptedException {
final Basket basket = new Basket(); // 定义一个producer
Runnable producer = new Runnable() {
public void run() {
try {
basket.produce();
} catch (InterruptedException ex) {
ex.printStackTrace();
}
}
}; // 定义一个consumer
Runnable consumer = new Runnable() {
public void run() {
try {
basket.consume();
} catch (InterruptedException ex) {
ex.printStackTrace();
}
}
}; // 各产生10个consumer和producer
ExecutorService service = Executors.newCachedThreadPool(); for (int i = 0; i < 4; i++)
service.submit(consumer); Thread.sleep(2000 * 2); for (int i = 0; i < 4; i++)
service.submit(producer); service.shutdown();
}
}

原因分析:

1、假设前面有2个producer(此时available=true)

1.1、一个在等待lock

1.2、一个await

2、consumer生成内容后,available=false,produced.signal(); 最后lock.unlock();

3.1、因为lock.unlock所以会触发一个lock获取到锁(虽然signal也会触发等待这个条件的其他线程,但是多线程大家都知道什么时候触发这是不确定的),如果此时正好是[1.1]那么因为available=false,执行完释放锁

3.2、produced.signal()所以会触发一个await的producer;

解决方案:

只要保证[3.1]还是需要await即可解决问题

所以加一个 AtomicInteger producedAwaitCounter = new AtomicInteger(0); 统计当前等待的生产者,如果当前available=false,但已经有生产者生成了内容,那么先等待消费者消费了再说

if (available || producedAwaitCounter.get() > 0) {

producedAwaitCounter.incrementAndGet();

produced.await(); // 放弃lock进入睡眠

producedAwaitCounter.decrementAndGet();

}

当然最简单的是使用:自旋,原理可以自己分析下:

while (available) {

produced.await(); // 放弃lock进入睡眠

}

  1. package com.sishuok.es.test;
  2. import java.util.concurrent.atomic.AtomicInteger;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.Lock;
  5. import java.util.concurrent.locks.ReentrantLock;
  6. //生产/消费者模式
  7. public class Basket {
  8. Lock lock = new ReentrantLock(true);
  9. // 产生Condition对象
  10. Condition produced = lock.newCondition();
  11. Condition consumed = lock.newCondition();
  12. boolean available = false;
  13. AtomicInteger producedAwaitCounter = new AtomicInteger();
  14. public void produce() throws InterruptedException {
  15. lock.lock();
  16. try {
  17. if (available || producedAwaitCounter.get() > ) {
  18. producedAwaitCounter.incrementAndGet();
  19. produced.await(); // 放弃lock进入睡眠
  20. producedAwaitCounter.decrementAndGet();
  21. }
  22. System.out.println("Apple produced.");
  23. available = true;
  24. consumed.signal(); // 发信号唤醒等待这个Condition的线程
  25. } finally {
  26. lock.unlock();
  27. }
  28. }
  29. public void consume() throws InterruptedException {
  30. lock.lock();
  31. try {
  32. if (!available) {
  33. consumed.await(); // 放弃lock进入睡眠
  34. }
  35. /* 吃苹果 */
  36. System.out.println("Apple consumed.");
  37. available = false;
  38. produced.signal(); // 发信号唤醒等待这个Condition的线程
  39. } finally {
  40. lock.unlock();
  41. }
  42. }
  43. }
package com.sishuok.es.test;

import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; //生产/消费者模式 public class Basket { Lock lock = new ReentrantLock(true); // 产生Condition对象 Condition produced = lock.newCondition(); Condition consumed = lock.newCondition(); boolean available = false;
AtomicInteger producedAwaitCounter = new AtomicInteger(0); public void produce() throws InterruptedException { lock.lock(); try { if (available || producedAwaitCounter.get() > 0) {
producedAwaitCounter.incrementAndGet();
produced.await(); // 放弃lock进入睡眠
producedAwaitCounter.decrementAndGet();
} System.out.println("Apple produced."); available = true; consumed.signal(); // 发信号唤醒等待这个Condition的线程 } finally {
lock.unlock();
} } public void consume() throws InterruptedException { lock.lock(); try { if (!available) {
consumed.await(); // 放弃lock进入睡眠
} /* 吃苹果 */ System.out.println("Apple consumed."); available = false; produced.signal(); // 发信号唤醒等待这个Condition的线程
} finally {
lock.unlock();
} } }

java生产者消费者问题代码分析的更多相关文章

  1. 基于Java 生产者消费者模式(详细分析)

    Java 生产者消费者模式详细分析 本文目录:1.等待.唤醒机制的原理2.Lock和Condition3.单生产者单消费者模式4.使用Lock和Condition实现单生产单消费模式5.多生产多消费模 ...

  2. Java 生产者消费者模式详细分析

    */ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...

  3. Java生产者消费者的三种实现

    Java生产者消费者是最基础的线程同步问题,java岗面试中还是很容易遇到的,之前没写过多线程的代码,面试中被问到很尬啊,面完回来恶补下.在网上查到大概有5种生产者消费者的写法,分别如下. 用sync ...

  4. Java 学习笔记 使用并发包ReentrantLock简化生产者消费者模式代码

    说明 ReentrantLock是java官方的一个线程锁类,ReentarntLock实现了Lock的接口 我们只需要使用这个,就可以不用使用synchronized同步关键字以及对应的notify ...

  5. java 生产者消费者问题 并发问题的解决

    引言 生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况: 生产者消费者图 ...

  6. JAVA生产者消费者的实现

    春节回了趟老家,又体验了一次流水席,由于桌席多,导致上菜慢,于是在等待间,总结了一下出菜流程的几个特点: 1.有多个灶台,多个灶台都在同时做菜出来. 2.做出来的菜,会有专人用一个托盘端出来,每次端出 ...

  7. Java生产者消费者模型

    在Java中线程同步的经典案例,不同线程对同一个对象同时进行多线程操作,为了保持线程安全,数据结果要是我们期望的结果. 生产者-消费者模型可以很好的解释这个现象:对于公共数据data,初始值为0,多个 ...

  8. java 生产者消费者问题 并发问题的解决(转)

    引言 生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况: 生产者消费者图 ...

  9. Linux 进程间通信(包含一个经典的生产者消费者实例代码)

    前言:编写多进程程序时,有时不可避免的需要在多个进程之间传递数据,我们知道,进程的用户的地址空间是独立,父进程中对数据的修改并不会反映到子进程中,但内核是共享的,大多数进程间通信方式都是在内核中建立一 ...

随机推荐

  1. 在ASP.NET中动态加载内容(用户控件和模板)

    在ASP.NET中动态加载内容(用户控件和模板) 要点: 1. 使用Page.ParseControl 2. 使用base.LoadControl 第一部分:加载模板 下 面是一个模板“<tab ...

  2. TCP粘包和半包的处理方法

    先把处理的方法的代码放这里: 解析数据帧的代码: bool CSocket::findData(byte* buff, int& len) { for (int i = 0; i <= ...

  3. javascript学习(9)——[设计模式]单例

    单例模式,相信大家对此都不陌生,我们主要讲下javascript中几个比较常见的设计模式: (1).普通的单体 (2).具有局部变量的强大单体 (3).惰性单体 (4).分支单体 下面我们就一一进行介 ...

  4. 我的Python成长之路---第七天---Python基础(22)---2016年2月27日(晴)

    socket网络编程 socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求. ...

  5. HYSBZ 2818 gcd

    /** 大意: 给定整数N,1<= x,y <= N 求解有多少gcd(x,y) 为素数 n=10^7 思路: 首先考虑到n 如此之大,用的快速求欧拉函数. 先默认 y〉x 分析: gcd ...

  6. hdoj 2222

    http://acm.hdu.edu.cn/showproblem.php?pid=2222 第一道 AC自动机.....trie树的建立 和 AC自动机的查询,,可作模版... 解题思路:AC的应用 ...

  7. Socket 基础解析使用ServerSocket建立聊天服务器

    很简单的教程哦! 1.socket 简介 Socket 又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求.ServerSocket 用于 ...

  8. jQuery入门学习贴

    先引一下阮一峰大牛翻译的书<软件随想录>中的一句话: “ joel认为,软件开发,最缺的是天才.而我认为,最缺的是全才.什么“全”:全面.全局.全心全意.全力以赴. ” 最近再学习jQue ...

  9. HDU p1294 Rooted Trees Problem 解题报告

    http://www.cnblogs.com/keam37/p/3639294.html keam所有 转载请注明出处 Problem Description Give you two definit ...

  10. 复习知识点:XML解析数据,JOSN解析数据,GET请求数据,POST请求数据

    GDataXMLDocument-XML解析 #pragma mark - ======== Dom XML解析 - (IBAction)domParserActionXML_Document:(id ...