JAVA 1.5 并发之 BlockingQueue
1.BlockingQueue 顾名思义就是阻塞队列
最经典的使用场合就是 生产者 - 消费者 模型啦,其优点是队列控制已经处理好,用户只需要存(满了会阻塞),取(空了会阻塞)
可以更多的关心核心逻辑,而不是并发控制
PS: BlockingDeque是阻塞双向队列,这里不作讲解,其实意思差不多啦
add(E e) 添加元素,如果失败则抛出exception
put(E e) 添加元素,如果队列满,则阻塞
offer(E e) 添加元素,如果队列满则返回false
offer(E e, long timeout, TimeUnit unit) 同上,但是有timeout
take(E e) 取出元素,如果队列空,则阻塞
poll( long timeout, TimeUnit unit) 在一定时间内取出元素,如无则返回null
drainTo() 一次性取出所有元素
2.各种阻塞队列
1) ArrayBlockingQueue
以数组为基础的queue, 建立时需要定义大小,同时可以定义公平锁 or not.
有一个特点是 元素在取出或者插入时使用同一个锁,或许是对性能影响不大?
2) LinkedBlockingQueue
以链表为基础的queue,建立时是不定义大小的,所以要注意生产太多内存会耗尽
与数组队列不同的是该队列在取出与写入是有分开的锁,理论上性能更好
但是具体还是得分情况,最好都试试,据说VM对array的优化很好,可能有惊喜
3) PriorityBlockingQueue
带有优先级的队列,同样不限定大小,只有在队列空的时候进行阻塞
定义的时候需要给与一个comparator
4)DelayQueue
延迟队列,具体使用情况,例如长时间无响应等
其实就是一个特别版的优先队列,优先度就是延迟,在获取元素的时候会根据延迟的大小决定是否返回元素
5)SynchronousQueue
同步队列,我的理解就是一个put对应一个take, 只有空的时候可以放东西,只能放一个
暂时没想到什么使用场合
3. 内部实现
具体细节我就不讲了,大家可以自己看,最主要就是在存取的时候用了之前说的lock来进行并发控制
而timeout则是由lock里condition的await实现,总体都不难,很容易看懂的
EXAMPLE
static ArrayBlockingQueue<Integer> arrayQueue;
static long startTime = System.currentTimeMillis(); public static void main(String[] args) {
arrayQueue = new ArrayBlockingQueue<Integer>(10); ExecutorService executor = Executors.newFixedThreadPool(4); //丢东西进去
Runnable producer = new Runnable() {
@Override
public void run() { for (int i = 0; i < 5; i++) {
try { int randomInt = i;
arrayQueue.put(randomInt); } catch (InterruptedException e) {
e.printStackTrace();
}
}
}
};
//拿东西出来
Runnable consumer = new Runnable() {
@Override
public void run() {
while (true)
{
try { System.out.println("remain capacity = "+arrayQueue.remainingCapacity());
Integer result = arrayQueue.poll(1,TimeUnit.SECONDS);
Thread.sleep(50);
if (result==null){
return;
} } catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}; //因为我的线程池是4,所以放4个任务进去
executor.submit(consumer);
executor.submit(producer);
executor.submit(producer);
executor.submit(producer); executor.shutdown(); }
OUTPUT:
remain capacity = 10
remain capacity = 0
remain capacity = 0
remain capacity = 0
remain capacity = 0
remain capacity = 0
remain capacity = 1
remain capacity = 2
remain capacity = 3
remain capacity = 4
remain capacity = 5
remain capacity = 6
remain capacity = 7
remain capacity = 8
remain capacity = 9
remain capacity = 10
JAVA 1.5 并发之 BlockingQueue的更多相关文章
- Java并发之BlockingQueue的使用
Java并发之BlockingQueue的使用 一.简介 前段时间看到有些朋友在网上发了一道面试题,题目的大意就是:有两个线程A,B, A线程每200ms就生成一个[0,100]之间的随机数, B线 ...
- java并发包——阻塞队列BlockingQueue及源码分析
一.摘要 BlockingQueue通常用于一个线程在生产对象,而另外一个线程在消费这些对象的场景,例如在线程池中,当运行的线程数目大于核心的线程数目时候,经常就会把新来的线程对象放到Blocking ...
- Java并发编程-阻塞队列(BlockingQueue)的实现原理
背景:总结JUC下面的阻塞队列的实现,很方便写生产者消费者模式. 常用操作方法 常用的实现类 ArrayBlockingQueue DelayQueue LinkedBlockingQueue Pri ...
- Java并发之BlockingQueue 阻塞队列(ArrayBlockingQueue、LinkedBlockingQueue、DelayQueue、PriorityBlockingQueue、SynchronousQueue)
package com.thread.test.thread; import java.util.Random; import java.util.concurrent.*; /** * Create ...
- Java并发之BlockingQueue
一.Queue Queue是队列接口是 Collection的子接口.除了基本的 Collection操作外,队列还提供其他的插入.提取和检查操作.每个方法都存在两种形式:一种抛出异常(操作失败时 ...
- JAVA多线程之间共享数据BlockingQueue介绍
在JAVA的Concurrent包中,BlockingQueue很好的解决了多线程中,如何高效安全“传输”数据的问题.通过这些高效并且线程安全的队列类,为我们快速搭建高质量的多线程程序带来极大的便利. ...
- java.util.concurrent 包笔记 --- BlockingQueue
BlockingQueue 队列接口,具有 4 组不同的方法用于插入.移除以及对队列中的元素进行检查.如果请求的操作不能得到立即执行的话,每个方法的表现也不同.这些方法如下: Throws exc ...
- java并发包分析之———BlockingQueue
一.概述: BlockingQueue作为线程容器,可以为线程同步提供有力的保障. 二.BlockingQueue定义的常用方法 1.BlockingQueue定义的常用方法如下: 抛出异常 ...
- Java并发编程——阻塞队列BlockingQueue
Java 并发编程系列文章 Java 并发基础——线程安全性 Java 并发编程——Callable+Future+FutureTask java 并发编程——Thread 源码重新学习 java并发 ...
随机推荐
- Bootstrap 第一天
Bootstrap第一天 1.什么是Bootstrap? Bootstrap是由两位设计开发的. Bootstrap主要是前端的框架(HTML.CSS.JS). 2.为什么使用Boot ...
- 关于ionic开发中遇到的坑与总结
这次是第二次使用ionic开发混合app,今天算是对这个框架做一个总结,基础的我就不再重复了,网上都有教程.我就说说自己的心得和遇见的各种坑, 之后会陆续补充,想到什么说什么吧. 1.关于ionic效 ...
- linux 4 -awk
十一. awk编程: 1. 变量: 在awk中变量无须定义即可使用,变量在赋值时即已经完成了定义.变量的类型可以是数字.字符串.根据使用的不同,未初始化变量的值为0或空白字符串&q ...
- struts2核心和工作原理
转至:http://blog.csdn.net/laner0515/article/details/27692673 在学习struts2之前,首先我们要明白使用struts2的目的是什么?它能给我们 ...
- 剑指offer——翻转单词顺序VS左旋转字符串
字符串的交换等,注意判断字符串的是否为NULL,以及判断边界等. #include <iostream> #include <string> using namespace s ...
- HDU 1800 Flying to the Mars 字典树,STL中的map ,哈希树
http://acm.hdu.edu.cn/showproblem.php?pid=1800 字典树 #include<iostream> #include<string.h> ...
- Spring Cloud之Swagger集群搭建
在微服务中,Swagger是每个服务 比如会员服务,订单服务,支付服务 进行继承. 如何将整个微服务中的Swagger进行合成,同一台服务器上. 使用Zuul+Swagger实现管理整个微服务API文 ...
- Spring- 异常org.xml.sax.SAXParseException; systemId: http://www.springframework.org/schema/context/; lineNumber: 1; columnNumber: 55; 在 publicId 和 systemId 之间需要有空格。
抛出异常 六月 03, 2018 7:40:44 下午 org.springframework.context.support.AbstractApplicationContext prepareRe ...
- spring学习(4)
在spring容器内拼凑bean叫做装配.装配bean的时候,需要告诉容器哪些bean以及容器如何使用依赖注入将它们配合在一起. 上下文定义文件的根元素是<beans>,<beans ...
- centos虚拟机启用网卡
CentOS虚拟机安装成功后,默认开机未启用网关,通过修改配置文件,启用网卡 编辑系统配置文件,虚拟机完成后,系统安装了一个默认的网卡,即eth0,其配置文件的路径为/etc/sysconfig/ne ...