博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
使用Java的BlockingQueue实现生产者-消费者
阅读量:4311 次
发布时间:2019-06-06

本文共 3241 字,大约阅读时间需要 10 分钟。

BlockingQueue也是java.util.concurrent下的主要用来控制线程同步的工具, BlockingQueue的实现是线程安全的。

BlockingQueue有四个具体的实现类,根据不同需求,选择不同的实现类

1、ArrayBlockingQueue:一个由数组支持的有界阻塞队列,规定大小的BlockingQueue, 其构造函数必须带一个int参数来指明其大小.其所含的对象是以FIFO(先入先出)顺序排序的

2、LinkedBlockingQueue:大小不定的BlockingQueue, 若其构造函数带一个规定大小的参数, 生成的BlockingQueue有大小限制, 若不带大小参数, 所生成的BlockingQueue的大小由Integer.MAX_VALUE来决定.其所含的对象是以FIFO(先入先出)顺序排序的。

3、PriorityBlockingQueue:类似于LinkedBlockQueue, 但其所含对象的排序不是FIFO, 而是依据对象的自然排序顺序或者是构造函数的Comparator决定的顺序。

4、SynchronousQueue:特殊的BlockingQueue, 对其的操作必须是放和取交替完成的。

 

LinkedBlockingQueue 可以指定容量,也可以不指定,不指定的话,默认最大是Integer.MAX_VALUE,其中主要用到put和take方法,put方法在队列满的时候会阻塞直到有队列成员被消费,take方法在队列空的时候会阻塞,直到有队列成员被放进来。

还有使用的一个方法是offer, offer放在插入一条数据到队列中,如果队满了会抛出一个异常并返回,offer的好处是不会阻塞。

 

生产者消费者的示例代码:

生产者:

import java.util.concurrent.BlockingQueue;    public class Producer implements Runnable {      BlockingQueue
queue; public Producer(BlockingQueue
queue) { this.queue = queue; } @Override public void run() { try { String temp = "A Product, 生产线程:" + Thread.currentThread().getName(); System.out.println("I have made a product:" + Thread.currentThread().getName()); queue.put(temp);//如果队列是满的话,会阻塞当前线程 } catch (InterruptedException e) { e.printStackTrace(); } } }

 

 消费者:

import java.util.concurrent.BlockingQueue;    public class Consumer implements Runnable{      BlockingQueue
queue; public Consumer(BlockingQueue
queue){ this.queue = queue; } @Override public void run() { try { String temp = queue.take();//如果队列为空,会阻塞当前线程 System.out.println(temp); } catch (InterruptedException e) { e.printStackTrace(); } } }

 

 测试类:

 

import java.util.concurrent.ArrayBlockingQueue;  import java.util.concurrent.BlockingQueue;  import java.util.concurrent.LinkedBlockingQueue;    public class Test3 {        public static void main(String[] args) {          BlockingQueue
queue = new LinkedBlockingQueue
(2); // BlockingQueue
queue = new LinkedBlockingQueue
(); //不设置的话,LinkedBlockingQueue默认大小为Integer.MAX_VALUE // BlockingQueue
queue = new ArrayBlockingQueue
(2); Consumer consumer = new Consumer(queue); Producer producer = new Producer(queue); for (int i = 0; i < 5; i++) { new Thread(producer, "Producer" + (i + 1)).start(); new Thread(consumer, "Consumer" + (i + 1)).start(); } } }

 

 打印结果:

I have made a product:Producer1  I have made a product:Producer2  A Product, 生产线程:Producer1  A Product, 生产线程:Producer2  I have made a product:Producer3  A Product, 生产线程:Producer3  I have made a product:Producer5  I have made a product:Producer4  A Product, 生产线程:Producer5  A Product, 生产线程:Producer4

 

由于队列的大小限定成了2,所以最多只有两个产品被加入到队列当中,而且消费者取到产品的顺序也是按照生产的先后顺序,原因就是LinkedBlockingQueue和ArrayBlockingQueue都是按照FIFO的顺序存取元素的。

  

转载于:https://www.cnblogs.com/davygeek/p/5695736.html

你可能感兴趣的文章
vnpy学习_04回测评价指标的缺陷
查看>>
iOS开发中遇到的问题整理 (一)
查看>>
Linux(SUSE 12)安装jboss4并实现远程访问
查看>>
Neutron在给虚拟机分配网络时,底层是如何实现的?
查看>>
netfilter/iptables全攻略
查看>>
Overlay之VXLAN架构
查看>>
Eclipse : An error occurred while filtering resources(Maven错误提示)
查看>>
在eclipse上用tomcat部署项目404解决方案
查看>>
web.xml 配置中classpath: 与classpath*:的区别
查看>>
suse如何修改ssh端口为2222?
查看>>
详细理解“>/dev/null 2>&1”
查看>>
suse如何创建定时任务?
查看>>
suse搭建ftp服务器方法
查看>>
centos虚拟机设置共享文件夹并通过我的电脑访问[增加smbd端口修改]
查看>>
文件拷贝(IFileOperation::CopyItem)
查看>>
MapReduce的 Speculative Execution机制
查看>>
大数据学习之路------借助HDP SANDBOX开始学习
查看>>
Hadoop基础学习:基于Hortonworks HDP
查看>>
为什么linux安装程序 都要放到/usr/local目录下
查看>>
Hive安装前扫盲之Derby和Metastore
查看>>