百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

Java集合之ArrayBlockingQueue java中集合retainall

liebian365 2024-11-12 13:10 5 浏览 0 评论

基于JDK 1.8 版本

概念

ArrayBlockingQueue是基于数组实现的阻塞队列,通过先进先出的顺序来访问元素。ArrayBlockingQueue是一个固定大小的有界队列,一旦创建,它的容量就不能再变化。如果向一个已经满的ArrayBlockingQueue中添加元素,则会导致该操作阻塞,直到该操作成功为止,同样的,如果从一个空的ArrayBlockingQueue中获取元素,也会导致阻塞。

使用

public static void main(String[] args) throws InterruptedException {
    // 创建一个容量为10的对象
    ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue<String>(10);
    // 初始化数据
    for (int i = 0; i < 10; i++) {
        arrayBlockingQueue.put(String.valueOf(i));
    }
    // 创建一个消费线程
    Thread consumer = new Thread(() -> {
        String value = "";
        try {
            // 等待10秒,降低消费频率
            Thread.sleep(10000L);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        while ((value = arrayBlockingQueue.poll()) != null) {
            System.out.println(value);
        }
    });

    consumer.start();
    // 主线程在arrayBlockingQueue满的情况下,再增加数据
    long now = System.currentTimeMillis();
    System.out.println("put 第11个数据,当前时间:" + now);
    arrayBlockingQueue.put("aaaa");
    System.out.println("put 第11个数据,花费:" + (System.currentTimeMillis() - now));
}

该段代码的逻辑是首先创建一个ArrayBlockingQueue对象,容量设置为10,并初始化数据。开启一个消费线程来消费该队列的数据,同时主线程在队列满的情况下,继续添加元素,为了达到等待效果,消费线程休眠10s钟,这时可以看到主线程在调用put方法时,会被阻塞。具体的执行结果如下:

分析

首先看下ArrayBlockingQueue的类图:

ArrayBlockingQueue实现了 Queue、Collection接口,所以它既有集合,也有队列相关的行为。

数据结构

    /** The queued items */
    final Object[] items; // 队列数组

    /** items index for next take, poll, peek or remove */
    int takeIndex; // 当前出队列的索引

    /** items index for next put, offer, or add */
    int putIndex; // 当前入队列的索引

    /** Number of elements in the queue */
    int count; // 当前队列的数量


    /** Main lock guarding all access */
    final ReentrantLock lock; // 并发锁

    /** Condition for waiting takes */
    private final Condition notEmpty; // 队列不空的信号量

    /** Condition for waiting puts */
    private final Condition notFull; // 队列不满的信号量

构造函数

    public ArrayBlockingQueue(int capacity, boolean fair) {
        if (capacity <= 0)
            throw new IllegalArgumentException();
        this.items = new Object[capacity];
        // 根据传入的参数,来决定是否创建公平锁
        lock = new ReentrantLock(fair);
        // 同一个锁创建的两个condition
        notEmpty = lock.newCondition();
        notFull =  lock.newCondition();
    }

在 ArrayBlockingQueue数据结构中,ReentrantLock 创建了两个 Condition对象:

notEmpty: 当队列不空的时候,会触发信号量,通知等待的线程

notFull: 当队列未满时,会触发信号量,通知等待的线程

注: Condition对象,是由指定的Lock对象创建, 当调用 Condition的await 方法时,会自动释放lock锁,并将当前线程阻塞,直到有其他线程调用该Condition的signal 方法、signalAll 方法、或是其他线程调用当前线程的interupt 方法,在await 方法返回当前线程时,当前线程必须重新获取和该Condition关联的锁。

put 操作

    public void put(E e) throws InterruptedException {
        checkNotNull(e);
        // 可重入锁
        final ReentrantLock lock = this.lock;
        // 获取锁
        lock.lockInterruptibly();
        try {
            // 如果当前已经满了,则等待信号量
            while (count == items.length)
                notFull.await();// 这里会释放锁,线程阻塞等待
            // 如果成功,则元素入队
            enqueue(e);
        } finally {
            // 解锁成功
            lock.unlock();
        }
    }

    private void enqueue(E x) {
        // assert lock.getHoldCount() == 1;
        // assert items[putIndex] == null;
        final Object[] items = this.items;
        items[putIndex] = x;
        if (++putIndex == items.length)
            putIndex = 0;
        count++;
        // notEmpty的信号量通知
        notEmpty.signal();
    }

put方法处理:

  1. 获取锁
  2. 如果获取锁成功,判断当前的元素数量是否满了,如果未满,直接加入队列,否则等待未满的信号量
  3. 释放锁

poll 方法

    public E poll() {
        final ReentrantLock lock = this.lock;
        // 获取锁
        lock.lock();
        try {
            // 如果数量为空,则返回null,否则调用dequeue
            return (count == 0) ? null : dequeue();
        } finally {
            lock.unlock();
        }
    }

    private E dequeue() {
        // assert lock.getHoldCount() == 1;
        // assert items[takeIndex] != null;
        final Object[] items = this.items;
        @SuppressWarnings("unchecked")
        E x = (E) items[takeIndex];
        // 返回taskIndex处的数据
        items[takeIndex] = null;
        if (++takeIndex == items.length)
            takeIndex = 0;
        count--;
        if (itrs != null)
            itrs.elementDequeued();
        // notFull的信号量通知
        notFull.signal();
        return x;
    }

处理逻辑:

  1. 获取锁
  2. 如果元素为空,返回 null, 如果不为空,返回takeIndex处理的元素,并处理notFull的信息量
  3. 释放锁

lock.lock() 与 lock.lockInterruptibly 的区别

在分析 offer与 put方法时,发现使用的锁定的方式是不同的

    public boolean offer(E e) {
        checkNotNull(e);
        final ReentrantLock lock = this.lock;
         // lock 
        lock.lock();
        try {
         ...
        } finally {
            lock.unlock();
        }
    }

    public void put(E e) throws InterruptedException {
        checkNotNull(e);
        final ReentrantLock lock = this.lock;
        lock.lockInterruptibly();
        try {
           ...
        } finally {
            lock.unlock();
        }
    }

lock.lock()与 lock.lockInterruptibly()的差异在于,调用锁定的线程是否允许被其他线程调用该线程的interupt方法,通常情况下,当锁被其他线程持有时,另一个线程去申请时,会被阻塞,一直等待在那边,如 synchronized 关键字、lock.lock() 方法等, 但当调用 lock.lockInterruptibly() 方法时,如果线程被其他的线程调用了interupt 方法时,则可以使被阻塞的线程跳出阻塞等待的逻辑。

相关推荐

快递查询教程,批量查询物流,一键管理快递

作为商家,每天需要查询许许多多的快递单号,面对不同的快递公司,有没有简单一点的物流查询方法呢?小编的回答当然是有的,下面随小编一起来试试这个新技巧。需要哪些工具?安装一个快递批量查询高手快递单号怎么快...

一键自动查询所有快递的物流信息 支持圆通、韵达等多家快递

对于各位商家来说拥有一个好的快递软件,能够有效的提高自己的工作效率,在管理快递单号的时候都需要对单号进行表格整理,那怎么样能够快速的查询所有单号信息,并自动生成表格呢?1、其实方法很简单,我们不需要一...

快递查询单号查询,怎么查物流到哪了

输入单号怎么查快递到哪里去了呢?今天小编给大家分享一个新的技巧,它支持多家快递,一次能查询多个单号物流,还可对查询到的物流进行分析、筛选以及导出,下面一起来试试。需要哪些工具?安装一个快递批量查询高手...

3分钟查询物流,教你一键批量查询全部物流信息

很多朋友在问,如何在短时间内把单号的物流信息查询出来,查询完成后筛选已签收件、筛选未签收件,今天小编就分享一款物流查询神器,感兴趣的朋友接着往下看。第一步,运行【快递批量查询高手】在主界面中点击【添...

快递单号查询,一次性查询全部物流信息

现在各种快递的查询方式,各有各的好,各有各的劣,总的来说,还是有比较方便的。今天小编就给大家分享一个新的技巧,支持多家快递,一次能查询多个单号的物流,还能对查询到的物流进行分析、筛选以及导出,下面一起...

快递查询工具,批量查询多个快递快递单号的物流状态、签收时间

最近有朋友在问,怎么快速查询单号的物流信息呢?除了官网,还有没有更简单的方法呢?小编的回答当然是有的,下面一起来看看。需要哪些工具?安装一个快递批量查询高手多个京东的快递单号怎么快速查询?进入快递批量...

快递查询软件,自动识别查询快递单号查询方法

当你拥有多个快递单号的时候,该如何快速查询物流信息?比如单号没有快递公司时,又该如何自动识别再去查询呢?不知道如何操作的宝贝们,下面随小编一起来试试。需要哪些工具?安装一个快递批量查询高手快递单号若干...

教你怎样查询快递查询单号并保存物流信息

商家发货,快递揽收后,一般会直接手动复制到官网上一个个查询物流,那么久而久之,就会觉得查询变得特别繁琐,今天小编给大家分享一个新的技巧,下面一起来试试。教程之前,我们来预览一下用快递批量查询高手...

简单几步骤查询所有快递物流信息

在高峰期订单量大的时候,可能需要一双手当十双手去查询快递物流,但是由于逐一去查询,效率极低,追踪困难。那么今天小编给大家分享一个新的技巧,一次能查询多个快递单号的物流,下面一起来学习一下,希望能给大家...

物流单号查询,如何查询快递信息,按最后更新时间搜索需要的单号

最近有很多朋友在问,如何通过快递单号查询物流信息,并按最后更新时间搜索出需要的单号呢?下面随小编一起来试试吧。需要哪些工具?安装一个快递批量查询高手快递单号若干怎么快速查询?运行【快递批量查询高手】...

连续保存新单号功能解析,导入单号查询并自动识别批量查快递信息

快递查询已经成为我们日常生活中不可或缺的一部分。然而,面对海量的快递单号,如何高效、准确地查询每一个快递的物流信息,成为了许多人头疼的问题。幸运的是,随着科技的进步,一款名为“快递批量查询高手”的软件...

快递查询教程,快递单号查询,筛选更新量为1的单号

最近有很多朋友在问,怎么快速查询快递单号的物流,并筛选出更新量为1的单号呢?今天小编给大家分享一个新方法,一起来试试吧。需要哪些工具?安装一个快递批量查询高手多个快递单号怎么快速查询?运行【快递批量查...

掌握批量查询快递动态的技巧,一键查找无信息记录的两种方法解析

在快节奏的商业环境中,高效的物流查询是确保业务顺畅运行的关键。作为快递查询达人,我深知时间的宝贵,因此,今天我将向大家介绍一款强大的工具——快递批量查询高手软件。这款软件能够帮助你批量查询快递动态,一...

从复杂到简单的单号查询,一键清除单号中的符号并批量查快递信息

在繁忙的商务与日常生活中,快递查询已成为不可或缺的一环。然而,面对海量的单号,逐一查询不仅耗时费力,还容易出错。现在,有了快递批量查询高手软件,一切变得简单明了。只需一键,即可搞定单号查询,一键处理单...

物流单号查询,在哪里查询快递

如果在快递单号多的情况,你还在一个个复制粘贴到官网上手动查询,是一件非常麻烦的事情。于是乎今天小编给大家分享一个新的技巧,下面一起来试试。需要哪些工具?安装一个快递批量查询高手快递单号怎么快速查询?...

取消回复欢迎 发表评论: