BlockingQueue

java.util.Concurrent包中,BlockingQueue很好的解决了在多线程中,如何高效安全“传输”数据的问题。通过这些高效并且线程安全的队列类,为我们快速搭建高质量的多线程程序带来极大的便利。同时,BlockingQueue也用于java自带线程池的缓冲队列中,了解BlockingQueue也有助于理解线程池的工作模型。

一 BlockingQueue接口

该接口属于队列,所以继承了Queue接口,该接口最重要的五个方法分别是offer方法poll方法put方法take方法drainTo方法

offer方法和poll方法分别有一个静态重载方法,分别是offer(E e, long timeout, TimeUnit unit)和poll(long timeout, TimeUnit unit)方法。其意义是在限定时间内存入或取出对象,如果不能存入取出则返回false。

put方法会在当队列存储对象达到限定值时阻塞线程,而在队列不为空时唤醒被take方法所阻塞的线程。take方法是相反的。

drainTo方法可批量获取队列中的元素。

二 常见的BlockingQueue实现

一 LinkedBlockingQueue

LinkedBlockingQueue是比较常见的BlockingQueue的实现,他是基于链表的阻塞队列。在创建该对象时如果不指定可存储对象个数大小时,默认为Integer.MAX_VALUE。当生产者往队列中放入一个数据时,队列会从生产者手中获取数据,并缓存在队列内部,而生产者立即返回;只有当队列缓冲区达到最大值缓存容量时,才会阻塞生产者队列,直到消费者从队列中消费掉一份数据,生产者线程会被唤醒,反之对于消费者这端的处理也基于同样的原理。

LinkedBlockingQueue内部使用了独立的两把锁来控制数据同步,这也意味着在高并发的情况下生产者和消费者可以并行地操作队列中的数据,以此来提高整个队列的并发性能。

put方法和offer方法:

public void put(E e) throws InterruptedException {
        if (e == null) throw new NullPointerException();
        int c = -1;
        Node<E> node = new Node(e);
        final ReentrantLock putLock = this.putLock;
        final AtomicInteger count = this.count;
        putLock.lockInterruptibly();
        try {
            while (count.get() == capacity) {
                notFull.await();
            }
            enqueue(node);
            c = count.getAndIncrement();
            if (c + 1 < capacity)
                notFull.signal();
        } finally {
            putLock.unlock();
        }
        if (c == 0)
            signalNotEmpty();
    }
 
    public boolean offer(E e) {
        if (e == null) throw new NullPointerException();
        final AtomicInteger count = this.count;
        if (count.get() == capacity)
            return false;
        int c = -1;
        Node<E> node = new Node(e);
        final ReentrantLock putLock = this.putLock;
        putLock.lock();
        try {
            if (count.get() < capacity) {
                enqueue(node);
                c = count.getAndIncrement();
                if (c + 1 < capacity)
                    notFull.signal();
            }
        } finally {
            putLock.unlock();
        }
        if (c == 0)
            signalNotEmpty();
        return c >= 0;
    }
}

这两个方法的区别是put方法在容量达到上限时会阻塞,而offer方法则会直接返回false。

二 ArrayBlockingQueue

ArrayBlockingQueue基于数组的阻塞队列,除了有一个定长数组外,ArrayBlockingQueue内部还保存着两个整形变量,分别标识着队列的头部和尾部在数组中的位置。ArrayBlockingQueue在生产者放入数据和消费者获取数据,都是共用同一个锁对象,由此也意味着两者无法真正并行运行,这点尤其不同于LinkedBlockingQueue。
ArrayBlockingQueue和LinkedBlockingQueue间还有一个明显的不同之处在于,前者在插入或删除元素时不会产生或销毁任何额外的对象实例,而后者则会生成一个额外的Node对象。

三 SynchronousQueue

是一种没有缓冲的阻塞队列,在生产者put的同时必须要有一个消费者进行take,否则就会阻塞。声明一个SynchronousQueue有两种不同的方式。公平模式和非公平模式的区别:如果采用公平模式:SynchronousQueue会采用公平锁,并配合一个FIFO队列来阻塞多余的生产者和消费者,从而体系整体的公平策略;但如果是非公平模式(SynchronousQueue默认):SynchronousQueue采用非公平锁,同时配合一个LIFO队列来管理多余的生产者和消费者,而后一种模式,如果生产者和消费者的处理速度有差距,则很容易出现饥渴的情况,即可能有某些生产者或者是消费者的数据永远都得不到处理。

四 PriorityBlockingQueue和DelayQueue

PriorityBlockingQueue是基于优先级的阻塞队列,该队列不会阻塞生产者,只会阻塞消费者。

DelayQueue队列存储的对象只有指定的延迟时间到了才能被取出,该队列也不会阻塞生产者。

三 BlockingQueue的使用

在处理多线程生产者消费者问题时的演示代码:

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
 
/**
 * Created by gavin on 15-8-30.
 */
public class BlockingQueueTest {
 
    public static void main(String[] args)
    {
        BlockingQueue<String> queue = new ArrayBlockingQueue<String>(1000);
        Thread p1 = new Thread(new Producer(queue),"producer1");
        Thread p2 = new Thread(new Producer(queue),"producer2");
        Thread c1 = new Thread(new Consumer(queue),"consumer1");
        Thread c2 = new Thread(new Consumer(queue),"consumer2");
 
        p1.start();
        p2.start();
        c1.start();
        c2.start();
    }
}
 
 
 
class Producer implements Runnable{
    private BlockingQueue<String> queue;
 
    public Producer(BlockingQueue<String> queue) {
        this.queue = queue;
    }
 
    public void run() {
        int i = 0;
        while (!Thread.currentThread().isInterrupted())
        {
            try {
                queue.put(Thread.currentThread().getName()+" product "+i);
            } catch (InterruptedException e) {
                System.err.println(Thread.currentThread().getName() + " error");
            }
            i++;
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
 
            }
        }
    }
}
class Consumer implements Runnable{
    private BlockingQueue<String> queue;
 
    public Consumer(BlockingQueue<String> queue) {
        this.queue = queue;
    }
 
    public void run() {
        int i = 0;
        while (!Thread.currentThread().isInterrupted())
        {
            try {
                String str = queue.take();
                System.out.println(str);
            } catch (InterruptedException e) {
 
            }
            try {
                Thread.sleep(300);
            } catch (InterruptedException e) {
 
            }
        }
    }
}

四 总结

BlockingQueue在并发编程中扮演着重要的角色,既可以自己用来解决生产者消费者问题,也用于java自带线程池的缓冲队列。

### 关于 Java 中 BlockingQueue 的介绍 #### 定义与概述 `BlockingQueue` 是 Java 并发包 `java.util.concurrent` 提供的一个接口,用于支持在生产者-消费者模式下的线程安全队列操作。该接口扩展了 `Queue` 接口并增加了阻塞功能,在某些情况下当执行插入或删除元素的操作时如果无法立即完成,则会等待一段时间甚至无限期地等待直至条件满足。 #### 方法分类及其行为描述 针对不同的应用场景需求,`BlockingQueue` 设计了几类具有不同特性的方法来处理入队和出队的行为: - **抛出异常**: 当尝试向满的队列中添加新项或将从空队列读取数据时将会触发特定类型的异常。 - 插入: 使用 `add(e)` 尝试加入元素 e;若失败则抛出 `IllegalStateException` 或其他适当异常[^1]。 - 移除: 利用 `remove()` 取消最前面的一项;如果没有可用项目就抛出 NoSuchElementException[]^1]^。 - **返回特殊值 (null/false)** : 这些版本不会阻止调用者的进展而是通过返回 false 表明未成功的动作或是 null 来指示不存在的数据。 - 插入: 应用程序可以使用 `offer(e)` 添加对象到队尾;如果因为容量原因而未能成功则给出布尔型结果表示状态。 - 移除: 对应地有 `poll()` 函数用来获取头部节点的内容;一旦发现为空即刻反馈 null 值作为回应[^2]。 - **无期限阻塞** :此类函数会在必要条件下挂起当前进程直到能够顺利完成预期的任务为止。 - 插入: `put(e)` 实现了将指定实体放置于容器末端的功能;即使空间不足也会持续等待资源释放后再继续工作[^4]。 - 移除: 同样存在 `take()` ,它负责提取首个成员并且只有当确实拥有可访问条目时才会结束休眠状态。 - **限时等待** :允许设定最大超时期限以控制长时间停滞的风险。 - 插入: 用户可通过 `offer(Object o, long timeout, TimeUnit unit)` 设置最长容忍延迟秒数以便决定何时停止重试机制。 - 移除: 类似地提供了带有参数配置选项的 `poll(long timeout, TimeUnit unit)` 方便开发者灵活调整策略。 #### 示例代码展示如何创建及运用一个简单的 BlockingQueue 结构 下面是一个简单例子展示了怎样声明以及初始化一个固定大小的工作任务缓冲区,并演示了一些基本操作: ```java import java.util.concurrent.*; public class SimpleBlockingQueueExample { public static void main(String[] args) throws InterruptedException { // 创建一个容量为5的LinkedBlockingQueue实例 BlockingQueue<String> queue = new LinkedBlockingQueue<>(5); System.out.println("Adding elements..."); try{ // 放置多个字符串进入队列 for(int i=0;i<7;i++){ String message="Task "+i; if(!queue.offer(message)){ System.out.println("Failed to add task due to full capacity."); } } Thread.sleep(2000); //模拟一些时间过去 while (!queue.isEmpty()){ //取出并打印每一个消息 System.out.println(queue.poll()); } }catch(Exception ex){ ex.printStackTrace(); } } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值