SynchronousQueue:同步Queue,属于线程安全的BlockingQueue的一种,此队列设计的理念类似于"单工模式",对于每个put/offer操作,必须等待一个take/poll操作,类似于我们的现实生活中的"火把传递":一个火把传递地他人,需要2个人"触手可及"才行. 因为这种策略,最终导致队列中并没有一个真正的元素;这是一种pipleline思路的基于queue的"操作传递".

如果有生产者,没有消费者,生产者的数据是put不了SynchronousQueue中的。

  • void put(E o):向队列提交一个元素,阻塞直到其他线程take或者poll此元素.
  • boolean offer(E o):向队列中提交一个元素,如果此时有其他线程正在被take阻塞(即其他线程已准备接收)或者"碰巧"有poll操作,那么将返回true,否则返回false.
  • E take():获取并删除一个元素,阻塞直到有其他线程offer/put.
  • boolean poll():获取并删除一个元素,如果此时有其他线程正在被put阻塞(即其他线程提交元素正等待被接收)或者"碰巧"有offer操作,那么将返回true,否则返回false.
  • E peek():总会返回null,硬编码.

这个队列中,对我们有意义的操作时put/take,以及put/offer + take或者put/take + poll,对于无法进入队列的元素,需要有额外的"拒绝"策略支持.

SynchronousQueue经常用来,一端或者双端严格遵守"单工"(单工作者)模式的场景,队列的两个操作端分别是productor和consumer.常用于一个productor多个consumer的场景。

在ThreadPoolExecutor中,通过Executors创建的cachedThreadPool就是使用此类型队列.已确保,如果现有线程无法接收任务(offer失败),将会创建新的线程来执行.

因为SynchronousQueue没有存储功能,因此put和take会一直阻塞,直到有另一个线程已经准备好参与到交付过程中。仅当有足够多的消费者,并且总是有一个消费者准备好获取交付的工作时,才适合使用同步队列。

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。

具体代码:

/**
* SynchronoutQueue 必须有生产者消费者同时,不然会一直阻塞。
* 没有生产者和消费者,队列中不会出现数据。
*/
public class Main {
public static void main(String[] args) {
SynchronousQueue<Integer> synchronousQueue = new SynchronousQueue<Integer>(); //生产者,不管几个生产者,只会有一个产品进入队列
Producer producer = new Producer(synchronousQueue);
producer.start();
Producer producer1 = new Producer(synchronousQueue);
producer1.start();
Producer producer2 = new Producer(synchronousQueue);
producer2.start(); //消费者
Consumer consumer = new Consumer(synchronousQueue);
consumer.start();
}
}
/**
* 生产者
*/
public class Producer extends Thread{
private SynchronousQueue<Integer> sychronousQueue;
public Producer(SynchronousQueue<Integer> sychronousQueue){
this.sychronousQueue = sychronousQueue;
} @Override
public void run() {
while(true){
int random = new Random().nextInt(1000);
System.out.println(Thread.currentThread().getName()+",生产一个产品:"+random);
System.out.println(Thread.currentThread().getName()+",等待三秒后运输出去");
try {
Thread.sleep(3000);
System.out.println(Thread.currentThread().getName()+",产品加入队列");
sychronousQueue.put(random);
System.out.println(Thread.currentThread().getName()+",加入成功");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
/**
* 消费者
*/
public class Consumer extends Thread{
private SynchronousQueue<Integer> synchronousQueue;
public Consumer(SynchronousQueue<Integer> synchronousQueue){
this.synchronousQueue = synchronousQueue;
} @Override
public void run() {
while (true){
try {
System.out.println(Thread.currentThread().getName()+",消费者等待");
int random = synchronousQueue.take();
System.out.println(Thread.currentThread().getName()+",消费者消费了产品:"+random);
System.out.println(Thread.currentThread().getName()+",-----------------------------------------------");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}

结果:

Thread-2,生产一个产品:329
Thread-2,等待三秒后运输出去
Thread-1,生产一个产品:204
Thread-1,等待三秒后运输出去
Thread-0,生产一个产品:167
Thread-0,等待三秒后运输出去
Thread-3,消费者等待
Thread-1,产品加入队列
Thread-0,产品加入队列
Thread-2,产品加入队列
Thread-3,消费者消费了产品:204
Thread-3,-----------------------------------------------
Thread-3,消费者等待
Thread-1,加入成功
Thread-3,消费者消费了产品:329
Thread-1,生产一个产品:496
Thread-1,等待三秒后运输出去
Thread-2,加入成功
Thread-3,-----------------------------------------------
Thread-3,消费者等待
Thread-2,生产一个产品:272
Thread-2,等待三秒后运输出去
Thread-0,加入成功
Thread-0,生产一个产品:7
Thread-0,等待三秒后运输出去
Thread-3,消费者消费了产品:167
Thread-3,-----------------------------------------------
Thread-3,消费者等待
Thread-1,产品加入队列
Thread-1,加入成功
Thread-1,生产一个产品:804
Thread-3,消费者消费了产品:496
Thread-3,-----------------------------------------------
Thread-3,消费者等待
Thread-1,等待三秒后运输出去
Thread-2,产品加入队列
Thread-2,加入成功
Thread-3,消费者消费了产品:272
Thread-3,-----------------------------------------------
Thread-3,消费者等待
Thread-0,产品加入队列
Thread-2,生产一个产品:648
Thread-2,等待三秒后运输出去
Thread-0,加入成功
Thread-0,生产一个产品:596
Thread-0,等待三秒后运输出去
Thread-3,消费者消费了产品:7
。。。。。。。

从结果中可以看出如果已经生产但是还未消费的,那么会阻塞在生产一直等到消费才能生成下一个。

源码地址:https://github.com/qjm201000/concurrent_synchronousQueue.git

并发编程-concurrent指南-阻塞队列-同步队列SynchronousQueue的更多相关文章

  1. 并发编程-concurrent指南-阻塞双端队列BlockingDeque

    java.util.concurrent 包里的 BlockingDeque 接口表示一个线程安放入和提取实例的双端队列. BlockingDeque 类是一个双端队列,在不能够插入元素时,它将阻塞住 ...

  2. 并发编程-concurrent指南-阻塞双端队列-链阻塞双端队列LinkedBlockingDeque

    LinkedBlockingDeque是双向链表实现的阻塞队列.该阻塞队列同时支持FIFO和FILO两种操作方式,即可以从队列的头和尾同时操作(插入/删除): 在不能够插入元素时,它将阻塞住试图插入元 ...

  3. 并发编程-concurrent指南-阻塞队列-数组阻塞队列ArrayBlockingQueue

    ArrayBlockingQueue类是实现了BlockingQueue. ArrayBlockingQueue是一个有界的阻塞队列,其内部实现是将对象放在一个数组中. 放入元素方法: (1) add ...

  4. 并发编程-concurrent指南-阻塞队列-链表阻塞队列LinkedBlockingQueue

    LinkedBlockingQueue是一个基于链表的阻塞队列. 由于LinkedBlockingQueue实现是线程安全的,实现了先进先出等特性,是作为生产者消费者的首选. LinkedBlocki ...

  5. 并发编程-concurrent指南-阻塞队列-优先级的阻塞队列PriorityBlockingQueue

    PriorityBlockingQueue是一个支持优先级的无界阻塞队列. 它使用了和类 java.util.PriorityQueue 一样的排序规则.你无法向这个队列中插入 null 值. 所有插 ...

  6. 并发编程-concurrent指南-阻塞队列BlockingQueue

    阻塞队列BlockingQueue,java.util.concurrent下的BlockingQueue接口表示一个线程放入和提取实例的队列. 适用场景: BlockingQueue通常用于一个线程 ...

  7. 并发编程-concurrent指南-阻塞队列-延迟队列DelayQueue

    DelayQueue是一个无界的BlockingQueue,用于放置实现了Delayed接口的对象,其中的对象只能在其到期时才能从队列中取走.这种队列是有序的,即队头对象的延迟到期时间最长.注意:不能 ...

  8. 并发编程-concurrent指南-原子操作类-AtomicInteger

    在java并发编程中,会出现++,--等操作,但是这些不是原子性操作,这在线程安全上面就会出现相应的问题.因此java提供了相应类的原子性操作类. 1.AtomicInteger

  9. 并发编程-concurrent指南-线程池ExecutorService的实例

    1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? new Thread(new Runnable() { @Override public void run() { ...

随机推荐

  1. UVA 10869 - Brownie Points II(树阵)

    UVA 10869 - Brownie Points II 题目链接 题意:平面上n个点,两个人,第一个人先选一条经过点的垂直x轴的线.然后还有一个人在这条线上穿过的点选一点作垂直该直线的线,然后划分 ...

  2. 让你的sublime text写C代码 (sublime text 2 配置构建C开发环境)

    原则 1. 首先你要配置能够编译C++/C环境 2. window中配置该执行环境的环境变量,能够全局使用 3. sublime Text创建新的构建机制.并设置用改全局编译环境 具体过程 能够编译C ...

  3. Jenkins build失败条件

    在Jenkins 项目写了很多剧本.有时候,我发现脚本失败,但Jenkins运行成功. Jenkins无论是通过退出代码0比量build成功. 因此,newLISP在.您可以使用(exit)对于成功. ...

  4. 用curl访问HTTPS站点并登录(对HTTP返回的结果特别清楚)

    开发网站,少不了测试.现在的网站为了加强安全性,都启用了HTTPS协议.所谓HTTPS,也就是HTTP文本在SSL协议中传输.用curl命令行来测试HTTPS站点是个很有用的功能,写点脚本,就可以做功 ...

  5. postgresql && .net core 使用空间数据

    这里主要讲遇到的一些报错 增删改查 && 计算部分基本和sql server的空间数据操作一毛一样,感谢微软大大的倾情支持,直接看demo即可(- ̄▽ ̄)- 前往sql server ...

  6. MVC基架生成的Create视图

    @model MyMusicStore.Models.Album @{     ViewBag.Title = "Create"; } <h2>Create</h ...

  7. gnuradio companion 找不到第三方模块gr-osmosdr的问题

    我使用了来自Ettus的gnuradio软件包,之后安装了gr-osmosdr 以在gnuradio中调用RTL电视棒. 但是在gnuradio companion找不到来自rtlsdr-source ...

  8. Expression Blend学习二UI布局

    什么是布局? · Panels控件(其实就是容器控件) · 对内部的子控件提供了自动布局功能 · 可以在容器控件内继续添加容器控件(一个复杂的界面往往是多种容器控件嵌套而组成的) · 一些界面器控件也 ...

  9. Win8 Metro(C#)数字图像处理--2.66FloodFill算法

    原文:Win8 Metro(C#)数字图像处理--2.66FloodFill算法  [函数名称]   洪水填充算法函数 WriteableBitmap FloodfillProcess(Write ...

  10. 浅谈.NET(C#)与Windows用户账户信息的获取

    原文:浅谈.NET(C#)与Windows用户账户信息的获取 目录 1. 用户账户名称 - 使用Environment类 2. 用户账户信息 - 使用WindowsIdentity和IdentityR ...