Java并发之阻塞队列BlockingQueue —— 线程合作

前言

当试图向队列添加元素而队列已满, 或是想从队列移出元素而队列为空的时候, 阻塞队列(blocking queue ) 导致线程阻塞。在协调多个线程之间的合作时,阻塞队列是一个有用的工具。工作者线程可以周期性地将中间结果存储在阻塞队列中。其他的工作者线程移出中间结果并进一步加以修改。

方法正常动作特殊情况下的动作
add添加一个元素如果队列满,则抛出 IllegalStateException 异常
element返回队列的头元素如果队列空,抛出 NoSuchElementException 异常
offer添加一个元素并返回 true如果队列满,返回 false
peek返回队列的头元素如果队列空, 则返回 null
poll移出并返回队列的头元素如果队列空, 则返回 null
put添加一个元素如果队列满, 则阻塞
remove移出并返回头元素如果队列空, 则抛出 NoSuchElementException 异常
take移出并返回头元素如果队列空, 则阻塞

其中put和take是阻塞方式,会阻塞当前线程的执行。

poll和offer也可以指定阻塞时间来实现阻塞

//尝试在 100 毫秒的时间内在队列的尾部插人一个元素。如果成功返回 true ; 否则false
boolean success = q.offer(x, 100, TimeUnit.MILLISECONDS);
//尝试用 100 毫秒的时间移除队列的头元素;如果成功返回头元素,否则,达到在超时时,返回 null
Object head = q.poll(100, TimeUnit.MILLISECONDS)

BlockingQueue

LinkedBlockingQueue 构造一个带有指定的容量和公平性设置的阻塞队列。该队列用循环数组实现。

容量是没有上边界的,但是,也可以选择指定最大容量。

LinkedBlockingDeque  根据指定容量构建一个有限的阻塞队列或双向队列,用链表实现

是一个双端的版本

ArrayBlockingQueue 根据指定容量构建一个有限的阻塞队列或双向队列,用链表实现。

在构造时需要指定容量,并且有一个可选的参数来指定是否需 要公平性。若设置了公平参数, 则那么等待了最长时间的线程会优先得到处理。通常,公平性会降低性能,只有在确实非常需要时才使用它。该队列用循环数组实现。

PriorityBlockingQueue 构造一个无边界阻塞优先队列,用堆实现。

是一个带优先级的队列, 而不是先进先出队列。元素按照它们的优先级顺序被移出。该队列是没有容量上限,但是,如果队列是空的,取元素的操作会阻塞。用堆实现.

LinkedTransferQueue (1.7版本)  传输一个值, 或者尝试在给定的超时时间内传输这个值, 这个调用将阻塞,直到另一 个线程将元素删除。

LinkedTransferQueue 实现了TransferQueue接口。TransferQueue 接口,允许生产者线程等待, 直到消费者准备就绪可以接收一个元素。如果生产者调用 q.transfer(item); 这个调用会阻塞, 直到另一个线程将元素(item) 删除

SynchronousQueue

运行一个线程把对象交给另一个线程

SynchronousQueue是一个阻塞队列,其中每个一元素的插入操作必须等待另一个线程进行移除操作

用一张图来说明类之间的关系

实例

构造一个SynchronousQueue阻塞队列

public class SynchronousQueueTest {

	public static void main(String[] args) {
		//SynchronousQueue 一个阻塞队列,其中每个插入操作必须等待另一个线程执行相应的移除操作
		SynchronousQueue<Integer> queue = new SynchronousQueue<>();
		Thread t = new Thread(()->{
			System.out.println("prepare to take...");
			try {
				System.out.println("i take an element " + queue.take());
			} catch (Exception e) {
				e.printStackTrace();
			}
		});
		
		Thread t2 = new Thread(()->{
			System.out.println("prepare to put...");
			try {
				queue.put(1);
                System.out.println("has been put...");
			} catch (Exception e) {
				e.printStackTrace();
			}
		});
		t2.start();
		t.start();

	}

}

prepare to put...
prepare to take...
has been put...
i take an element 1

如果把线程一中 queue.take() 操作移除,线程二的queue.put(1)无法执行

执行结果如

prepare to take...
prepare to put...

(线程二总在执行...)

put和take使用的简单实例:

容量为1的阻塞队列,线程1会阻塞,直到线程2放入元素。

@Test
    public void test1() throws Exception{
        LinkedBlockingQueue queue = new LinkedBlockingQueue(1);
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(5,20,60, TimeUnit.SECONDS,new LinkedBlockingQueue(10));
        threadPoolExecutor.submit(()->{
            System.out.println("prepare to take..");
            try {
                int i =(int)queue.take();
                System.out.println("i get " + i);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        TimeUnit.SECONDS.sleep(3);
        threadPoolExecutor.submit(()->{
            System.out.println("prepare to put..");
            try {
                queue.put(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
    }

参考:java并发编程实战

          java核心技术卷I 第14章 (第10版)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值