SynchronousQueue应用
SynchronousQueue是无界的,是一种无缓冲的等待队列,但是由于该Queue本身的特性,在某次添加元素后必须等待其他线程取走后才能继续添加;可以认为SynchronousQueue是一个缓存值为1的阻塞队列,但是 isEmpty()方法永远返回是true,remainingCapacity() 方法永远返回是0,remove()和removeAll() 方法永远返回是false,iterator()方法永远返回空,peek()方法永远返回null。
声明一个SynchronousQueue有两种不同的方式,它们之间有着不太一样的行为。
公平模式和非公平模式的区别:如果采用公平模式:SynchronousQueue会采用公平锁,并配合一个FIFO队列来阻塞多余的生产者和消费者,从而体系整体的公平策略;
但如果是非公平模式(SynchronousQueue默认):SynchronousQueue采用非公平锁,同时配合一个LIFO队列来管理多余的生产者和消费者,而后一种模式,如果生产者和消费者的处理速度有差距,则很容易出现饥渴的情况,即可能有某些生产者或者是消费者的数据永远都得不到处理。
=======================================================================================
SynchronousQueue是这样 一种阻塞队列,其中每个 put 必须等待一个 take,反之亦然。同步队列没有任何内部容量,甚至连一个队列的容量都没有。
不能在同步队列上进行 peek,因为仅在试图要取得元素时,该元素才存在;
除非另一个线程试图移除某个元素,否则也不能(使用任何方法)添加元素;
也不能迭代队列,因为其中没有元素可用于迭代。队列的头是尝试添加到队列中的首个已排队线程元素; 如果没有已排队线程,则不添加元素并且头为 null。
注意1:它一种阻塞队列,其中每个 put 必须等待一个 take,反之亦然。
同步队列没有任何内部容量,甚至连一个队列的容量都没有。
注意2:它是线程安全的,是阻塞的。
注意3:不允许使用 null 元素。
注意4:公平排序策略是指调用put的线程之间,或take的线程之间。
公平排序策略可以查考ArrayBlockingQueue中的公平策略。
注意5:SynchronousQueue的以下方法很有趣:
* iterator() 永远返回空,因为里面没东西。
* peek() 永远返回null。
* put() 往queue放进去一个element以后就一直wait直到有其他thread进来把这个element取走。
* offer() 往queue里放一个element后立即返回,如果碰巧这个element被另一个thread取走了,offer方法返回true,认为offer成功;否则返回false。
* offer(2000, TimeUnit.SECONDS) 往queue里放一个element但是等待指定的时间后才返回,返回的逻辑和offer()方法一样。
* take() 取出并且remove掉queue里的element(认为是在queue里的。。。),取不到东西他会一直等。
* poll() 取出并且remove掉queue里的element(认为是在queue里的。。。),只有到碰巧另外一个线程正在往queue里offer数据或者put数据的时候,该方法才会取到东西。否则立即返回null。
* poll(2000, TimeUnit.SECONDS) 等待指定的时间然后取出并且remove掉queue里的element,其实就是再等其他的thread来往里塞。
* isEmpty()永远是true。
* remainingCapacity() 永远是0。
* remove()和removeAll() 永远是false。
Demo:
简化版:
import java.util.Random;
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.TimeUnit; public class Main {
public static void main(String[] args) throws InterruptedException {
SynchronousQueue<Integer> queue = new SynchronousQueue<Integer>();
new Customer(queue).start();
new Product(queue).start();
} static class Product extends Thread{
SynchronousQueue<Integer> queue;
public Product(SynchronousQueue<Integer> queue){
this.queue = queue;
}
@Override
public void run(){
while(true){
int rand = new Random().nextInt(1000);
System.out.println("生产了一个产品:"+rand);
System.out.println("等待三秒后运送出去...");
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
queue.offer(rand);
System.out.println("产品生成完成:"+rand);
}
}
}
static class Customer extends Thread{
SynchronousQueue<Integer> queue;
public Customer(SynchronousQueue<Integer> queue){
this.queue = queue;
}
@Override
public void run(){
while(true){
try {
System.out.println("消费了一个产品:"+queue.take());
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("------------------------------------------");
}
}
}
}
生产了一个产品:326
等待三秒后运送出去...
产品生成完成:326
生产了一个产品:291
等待三秒后运送出去...
消费了一个产品:326
------------------------------------------
产品生成完成:291
消费了一个产品:291
------------------------------------------
生产了一个产品:913
等待三秒后运送出去...
产品生成完成:913
消费了一个产品:913
------------------------------------------
生产了一个产品:993
等待三秒后运送出去...
产品生成完成:993
消费了一个产品:993
------------------------------------------
生产了一个产品:295
等待三秒后运送出去...
产品生成完成:295
消费了一个产品:295
------------------------------------------
生产了一个产品:772
等待三秒后运送出去...
产品生成完成:772
消费了一个产品:772
------------------------------------------
生产了一个产品:977
等待三秒后运送出去...
产品生成完成:977
消费了一个产品:977
------------------------------------------
生产了一个产品:182
等待三秒后运送出去...
产品生成完成:182
消费了一个产品:182
------------------------------------------
生产了一个产品:606
等待三秒后运送出去...
产品生成完成:606
消费了一个产品:606
------------------------------------------
生产了一个产品:704
等待三秒后运送出去...
产品生成完成:704
消费了一个产品:704
------------------------------------------
生产了一个产品:194
等待三秒后运送出去...
产品生成完成:194
生产了一个产品:355
等待三秒后运送出去...
消费了一个产品:194
------------------------------------------
产品生成完成:355
消费了一个产品:355
------------------------------------------
生产了一个产品:991
等待三秒后运送出去...
产品生成完成:991
消费了一个产品:991
------------------------------------------
生产了一个产品:958
等待三秒后运送出去...
产品生成完成:958
消费了一个产品:958
------------------------------------------
生产了一个产品:388
等待三秒后运送出去..
从结果中可以看出如果已经生产但是还未消费的,那么会阻塞在生产一直等到消费才能生成下一个。
多线程版本:
import java.util.Random;
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.TimeUnit; public class Main {
public static void main(String[] args) throws InterruptedException {
SynchronousQueue<String> queue=new SynchronousQueue();
// TODO Auto-generated method stub
for(int i=0;i<5;i++)
new Thread(new ThreadProducer(queue)).start();
for(int i=0;i<5;i++)
new Thread(new ThreadConsumer(queue)).start();
}
}
class ThreadProducer implements Runnable {
ThreadProducer(SynchronousQueue<String> queue)
{
this.queue=queue;
}
SynchronousQueue<String> queue;
static int cnt=0;
public void run()
{
String name="";
int val=0;
Random random =new Random(System.currentTimeMillis());
for(int i=0;i<2;i++){ cnt=(cnt+1)&0xFFFFFFFF; try{
val=random.nextInt()%15;
if(val<5)
{
name="offer name:"+cnt;
queue.offer(name);
}
else if(val<10)
{
name="put name:"+cnt;
queue.put(name);
}
else
{
name="offer wait time and name:"+cnt;
queue.offer(name, 1000, TimeUnit.MILLISECONDS);
}
Thread.sleep(1);
}catch(InterruptedException e)
{
e.printStackTrace();
}
}
}
} class ThreadConsumer implements Runnable {
ThreadConsumer(SynchronousQueue<String> queue) {
this.queue = queue;
}
SynchronousQueue<String> queue; public void run() {
String name;
for(int i=0;i<2;i++){
try {
name = queue.take();
System.out.println("take " + name);
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
take offer wait time and name:4
take offer wait time and name:4
take offer wait time and name:5
take offer wait time and name:4
take offer wait time and name:4
take offer name:9
结果有很多种可能性,要自己尝试运行。
http://blog.csdn.net/hudashi/article/details/7076814
http://ifeve.com/java-synchronousqueue/
http://blog.csdn.net/liu88010988/article/details/50789179
SynchronousQueue应用的更多相关文章
- 【实战Java高并发程序设计 7】让线程之间互相帮助--SynchronousQueue的实现
[实战Java高并发程序设计 1]Java中的指针:Unsafe类 [实战Java高并发程序设计 2]无锁的对象引用:AtomicReference [实战Java高并发程序设计 3]带有时间戳的对象 ...
- 【JUC】JDK1.8源码分析之SynchronousQueue(九)
一.前言 本篇是在分析Executors源码时,发现JUC集合框架中的一个重要类没有分析,SynchronousQueue,该类在线程池中的作用是非常明显的,所以很有必要单独拿出来分析一番,这对于之后 ...
- SynchronousQueue类
介绍 Java 6的并发编程包中的SynchronousQueue是一个没有数据缓冲的BlockingQueue,生产者线程对其的插入操作put必须等待消费者的移除操作take,反过来也一样.不像Ar ...
- Java并发之BlockingQueue 阻塞队列(ArrayBlockingQueue、LinkedBlockingQueue、DelayQueue、PriorityBlockingQueue、SynchronousQueue)
package com.thread.test.thread; import java.util.Random; import java.util.concurrent.*; /** * Create ...
- SynchronousQueue 的简单应用
SynchronousQueue是这样一种阻塞队列,其中每个 put 必须等待一个 take,反之亦然.同步队列没有任何内部容量,甚至连一个队列的容量都没有. 不能在同步队列上进行 peek ...
- java多线程之:SynchronousQueue队列
SynchronousQueue是这样一种阻塞队列,其中每个 put 必须等待一个 take,反之亦然.同步队列没有任何内部容量,甚至连一个队列的容量都没有. 不能在同步队列上进行 peek ...
- SynchronousQueue
SynchronousQueue是一个没有数据缓冲的BlockingQueue,生产者线程对其的插入操作put必须等待消费者的移除操作take,反过来也一样. 不像ArrayBlockingQueue ...
- 【转】简析SynchronousQueue,LinkedBlockingQueue,ArrayBlockingQueue
转载地址:http://blog.csdn.net/mn11201117/article/details/8671497 SynchronousQueue SynchronousQueue是无界的,是 ...
- java concurrent 之 SynchronousQueue
Java 6的并发编程包中的SynchronousQueue是一个没有数据缓冲的BlockingQueue,生产者线程对其的插入操作put必须等待消费者的移除操作take,反过来也一样. 不像Arra ...
随机推荐
- PE-2 & 暴模...
题意: 求不大于4000000的斐波那契数列中,所有偶数之和. SOL: 还是只会暴模...看讨论区貌似有一个很牛逼的大神的发言? 英语水平太差... mark以下 The Fibonacci seq ...
- ACM The Famous Clock
The Famous Clock 时间限制:1000 ms | 内存限制:65535 KB 难度:1 描述 Mr. B, Mr. G and Mr. M are now in Warsaw, ...
- [知识点]网络流之Edmond-Karp算法
// 此博文为迁移而来,写于2015年2月2日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102vr12.html ...
- Codeforces Beta Round #6 (Div. 2 Only)
A,B,C都是水题... D题,直接爆搜.我换了好多姿势,其实最简单的方法,就能过. #include <cstdio> #include <string> #include ...
- Redhat5.8 环境下编译安装 Redis 并将其注册为系统服务
系统环境: $ cat /etc/issueRed Hat Enterprise Linux Server release 5.8 (Tikanga)Kernel \r on an \m 1. 下载安 ...
- win7 、win10连接l2tpvpn
win7: 修改vpn连接选项: win10: 参考连接: http://service.njaf.gov.cn/26970/26971/201510/t20151024_3621861.html ...
- webform FileUpload控件实例应用 上传图片
首先在根目录下建一个"images"文件: HTML: <form id="form1" runat="server"> < ...
- c#语句 类
知识点: 1.string类 2.Math类 3.DateTime 获取时间 for穷举 1.羽毛球拍15元,球3元,水2元.现有200元,每种至少买一个,共有多少种可能.
- 【HDU4585 Shaolin】map的经典运用
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4585 题意大意:很多人想进少林寺,少林寺最开始只有一个和尚,每个人有有一个武力值,若这个人想进少林,必 ...
- 允许浏览器跨域访问web服务端的解决方案
今天和同事探讨了前后端如何真正实现隔离开发的问题,如果前端单独作为服务发布,势必会涉及到无法直接调用后端的接口的问题,因为浏览器是不允许跨域提交请求的. 所谓跨域访问,就是在浏览器窗口,和某个服务端通 ...