前面阐述了实现生产者与消费者问题的两种方式:wait() / notify()方法 和 await() / signal()方法,本文继续阐述多线程的经典问题---生产者与消费者的第三种方式:BlockingQueue阻塞队列方法

BlockingQueue阻塞队列方法

BlockingQueue是JDK5.0的新增内容,它是一个已经在内部实现了同步的队列,实现方式采用的是我们第2种await() / signal()方法。它可以在生成对象时指定容量大小。它用于阻塞操作的是put()和take()方法。

put()方法:类似于我们上面的生产者线程,容量达到最大时,自动阻塞。

take()方法:类似于我们上面的消费者线程,容量为0时,自动阻塞。

缓冲区(仓库):

import java.util.concurrent.LinkedBlockingQueue;

public class Storage {
// 仓库最大存储量
private final int MAX_SIZE = 100; // 仓库存储的载体
private LinkedBlockingQueue<Object> list = new LinkedBlockingQueue<Object>(100); // 生产num个产品
public void produce(int num){
// 如果仓库剩余容量为MAX_SIZE
if (list.size() == MAX_SIZE){
System.out.println("【库存量】:" + MAX_SIZE + "暂时不能执行生产任务!");
} // 生产条件满足情况下,生产num个产品
for (int i = 1; i <= num; ++i){
try{
// 放入产品,自动阻塞
list.put(new Object());
}
catch (InterruptedException e){
e.printStackTrace();
} System.out.println("【现仓储量为】:" + list.size());
}
} // 消费num个产品
public void consume(int num){
// 如果仓库存储量不足
if (list.size() == 0){
System.out.println("【库存量】:0 暂时不能执行消费任务!");
} // 消费条件满足情况下,消费num个产品
for (int i = 1; i <= num; ++i){
try{
// 消费产品,自动阻塞
list.take();
}
catch (InterruptedException e){
e.printStackTrace();
}
} System.out.println("【现仓储量为】:" + list.size());
} // set/get方法
public LinkedBlockingQueue<Object> getList()
{
return list;
} public void setList(LinkedBlockingQueue<Object> list)
{
this.list = list;
} public int getMAX_SIZE()
{
return MAX_SIZE;
}
}

生产者:

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();
}
}

生产者与消费者(三)---BlockingQueue的更多相关文章

  1. Spring MVC 使用介绍(七)—— 注解式控制器(三):生产者与消费者模型

    一.MIME类型 MIME类型格式:type/subtype(;parameter)? type:主类型,任意的字符串,如text,如果是*号代表所有 subtype:子类型,任意的字符串,如html ...

  2. C#多线程学习(三) 生产者和消费者

    前面说过,每个线程都有自己的资源,但是代码区是共享的,即每个线程都可以执行相同的函数.这可能带来的问题就是几个线程同时执行一个函数,导致数据的混乱,产生不可预料的结果,因此我们必须避免这种情况的发生. ...

  3. RabbitMQ学习笔记(三、生产者与消费者)

    目录: 细说交换器 细说队列 发送消息 消费消息 确认与拒绝 细说交换器: 1.方法: public AMQP.Exchange.DeclareOk exchangeDeclare(String ex ...

  4. 生产者与消费者(一)---wait与notify

    生产者消费者问题是研究多线程程序时绕不开的经典问题之一,它描述是有一块缓冲区作为仓库,生产者可以将产品放入仓库,消费者则可以从仓库中取走产品.解决生产者/消费者问题的方法可分为两类: (1)采用某种机 ...

  5. Java中生产者与消费者模式

    生产者消费者模式 首先来了解什么是生产者消费者模式.该模式也称有限缓冲问题(英语:Bounded-buffer problem),是一个多线程同步问题的经典案例.该问题描述了两个共享固定大小缓冲区的线 ...

  6. 【JUC】阻塞队列&生产者和消费者

    阻塞队列 线程1往阻塞队列添加元素[生产者] 线程2从阻塞队列取出元素[消费者] 当队列空时,获取元素的操作会被阻塞 当队列满时,添加元素的操作会被阻塞 阻塞队列的优势:在多线程领域,发生阻塞时,线程 ...

  7. LMAX Disruptor—多生产者多消费者中,消息复制分发的高性能实现

    解决的问题 当我们有多个消息的生产者线程,一个消费者线程时,他们之间如何进行高并发.线程安全的协调? 很简单,用一个队列. 当我们有多个消息的生产者线程,多个消费者线程,并且每一条消息需要被所有的消费 ...

  8. java多线程中的生产者与消费者之等待唤醒机制@Version1.0

    一.生产者消费者模式的学生类成员变量生产与消费demo,第一版1.等待唤醒:    Object类中提供了三个方法:    wait():等待    notify():唤醒单个线程    notify ...

  9. Android(java)学习笔记71:生产者和消费者之等待唤醒机制

    1. 首先我们根据梳理我们之前Android(java)学习笔记70中关于生产者和消费者程序思路: 2. 下面我们就要重点介绍这个等待唤醒机制: (1)第一步:还是先通过代码体现出等待唤醒机制 pac ...

随机推荐

  1. Eclipse&Spring开发开发环境配置

    下载Eclipse,打开http://www.eclipse.org/downloads/,选择J2EE开发版: 根据操作系统选择32位还是64位. 开发工具:SpringSource Tool Su ...

  2. Apache.NMS.Stomp 下载

    最近项目中有用到ActiveMQ, MQ服务器61613的端口是用的STOMP协议, 原来项目中有使用MQ, 但发现缺少Apache.NMS.Stomp.dll的引用,于是上官网上找,结果发现所有的A ...

  3. [App]华为P6设置与Xamarin Studio连通测试

    使用模拟器进行调试十分麻烦,而且速度很慢,手头上有手机一台正好做测试机器,不过一直无法连通电脑. 百度了一番才知道,首先要在连接时候选择: PC 助手(HiSuite) 这样会加载一个类似驱动盘的资源 ...

  4. hdu 4790 Just Random (2013成都J题) 数学思路题 容斥

    题意:在[a,b]  [c,d] 之间,和模p等于m的对数 详见代码 #include <stdio.h> #include <algorithm> #include < ...

  5. RGB Bayer Color分析

    RGB Bayer Color分析 Bayer色彩滤波阵列 拜耳色彩滤波阵列(Bayer Color Filter Array,CFA)是非常有名的彩色图片的数字采集格式.色彩滤波器的模式如上图所示, ...

  6. HW4.13

    public class Solution { public static void main(String[] args) { int n = 0; while(n * n * n < 120 ...

  7. yii快速入门与参考

    Yii 权威指南 http://www.yiichina.com/guide/ http://www.yiiframework.com/doc/guide/1.1/zh_cn/ Ⅰ.基本概念一.入口文 ...

  8. /etc/init.d/syslog: No such file or directory

    centos 6 使用的是 rsyslog 而 centos5 使用 syslog ,兩個不同版本使用軟體不同.因此,你得要修訂 /etc/rsyslog.conf 才行!軟體也變成 /etc/ini ...

  9. poj 3608 Bridge Across Islands

    题目:计算两个不相交凸多边形间的最小距离. 分析:计算几何.凸包.旋转卡壳.分别求出凸包,利用旋转卡壳求出对踵点对,枚举距离即可. 注意:1.利用向量法判断旋转,而不是计算角度:避免精度问题和TLE. ...

  10. 使用VNC实现多用户登录linux系统

    vmare版本:12.0.0 build-2985596