Java 入门指南:Java 并发编程 —— 并发容器 PriorityBlockingQueue

news2025/2/24 2:32:42

BlockingQueue

BlockingQueue 是Java并发包(java.util.concurrent)中提供的一个阻塞队列接口,它继承自 Queue 接口。

BlockingQueue 中的元素采用 FIFO 的原则,支持多线程环境并发访问,提供了阻塞读取和写入的操作,当前线程在队列满或空的情况下会被阻塞,直到被唤醒或超时。

常用的实现类有:

  • ArrayBlockingQueue:并发容器 ArrayBlockingQueue 详解
  • LinkedBlockingQueue:并发容器 LinkedBlockingQueue 详解
  • PriorityBlockingQueue
  • SynchronousQueue:并发容器 SynchronousQueue 详解
  • LinkedBlockingDeque:并发容器 LinkedBlockingDeque 详解
    等类,它们的实现方式各有不同。
适用场景

BlockingQueue 通常用于一个线程生产对象,而另外一个线程消费这些对象的场景。

![[BlockingQueue Model.png]]

一个线程将会持续生产新对象并将其插入到队列之中,直到队列达到它所能容纳的临界点

如果该阻塞队列到达了其临界点,生产者线程将会在往里边插入新对象时发生阻塞。它会一直处于阻塞之中,直到消费者线程从队列中拿走一个对象。

消费者线程将会一直从该阻塞队列中拿出对象。如果消费线程尝试去从一个空的队列中提取对象的话,这个消费线程将会处于阻塞之中,直到一个生产线程把一个对象丢进队列。

常用方法
  1. put(E e):将元素 e 插入到队列中,如果队列已满,则会阻塞当前线程,直到队列有空闲空间

  2. offer(E e):将元素 e 插入到队列中,如果队列已满,则返回 false。

  3. offer(E element, long timeout, TimeUnit unit) 方法是 BlockingQueue:在指定的时间内将元素添加到队列中。

    • timeout:超时时间,表示在指定的时间内等待队列空间可用。如果超过指定的时间仍然无法将元素添加到队列中,将返回 false。

    • unit:超时时间的单位。

  4. take():移除并返回队列头部的元素,如果队列为空,则会阻塞当前线程,直到队列有元素

  5. poll():移除并返回队列头部的元素,如果队列为空,则返回 null

  6. poll(long timeout, TimeUnit unit):在指定的时间内从队列中检索并移除元素。返回移除的元素。如果超过指定的时间仍然没有可用的元素,将返回 null。

  7. peek():返回队列头部的元素,但不会移除。如果队列为空,则返回null

  8. size():返回队列中元素的数量

  9. isEmpty():判断队列是否为空,为空返回 true,否则返回 false

  10. isFull():判断队列是否已满,已满返回 true,否则返回 false

  11. clear():清空队列中的所有元素

行为抛异常返回特定值阻塞超时
插入add(o)offer(o)put(o)offer(o, timeout, timeunit)
移除remove()poll()take()poll(timeout, timeunit)
检查element()peek()
  • 抛异常: 如果试图的操作无法立即执行,抛一个异常。

  • 特定值: 如果试图的操作无法立即执行,返回 true / false / null)。

  • 阻塞: 如果试图的操作无法立即执行,该方法调用将会发生阻塞,直到能够执行。

  • 超时: 如果试图的操作无法立即执行,该方法调用将会发生阻塞,直到能够执行,但等待时间不会超过给定值。返回一个特定值以告知该操作是否成功(典型的是 true / false)。

若向 BlockingQueue 中插入 null,将会抛出 NullPointerException

死锁问题

需要注意的是,在使用 BlockingQueue 时要注意防止死锁的问题:

  • 在队列满之后调用 offer() 方法插入元素会返回false,此时不能直接调用put()方法,因为在插入之前还需要获取其它资源,如果在获取资源时一直阻塞在这里,就会发生死锁。

  • 为了防止死锁的问题,建议使用 offer(E e, long timeout, TimeUnit unit)poll(long timeout, TimeUnit unit) 带有超时时间的方法。

PriorityBlockingQueue

PriorityBlockingQueue 是一个无界的并发队列。它使用了和类PriorityQueue 一样的排序规则。

所有插入到 PriorityBlockingQueue 的元素必须实现 java.lang.Comparable 接口。因此该队列中元素的排序就取决于自定义的 Comparable 实现。如果元素没有实现 Comparable 接口并且没有提供自定义的比较器,将会引发 ClassCastException

PriorityBlockingQueue 对于具有相等优先级(compare() == 0)的元素并不强制任何特定行为。

特点
  • 线程安全PriorityBlockingQueue是线程安全的,它使用锁(如ReentrantLock)来确保在多线程环境下的数据一致性和操作的原子性。

  • 优先级排序:队列中的元素会根据它们的优先级进行排序,默认情况下,元素按照其自然顺序(如果实现了Comparable接口)进行排序,或者根据构造时提供的Comparator进行排序。

  • 阻塞特性:当队列为空时,尝试从队列中取出元素的线程会被阻塞,直到队列中有元素可用;当队列已满(对于无界队列,实际上是系统资源耗尽)时,尝试向队列中添加元素的线程也会被阻塞,或者可以选择非阻塞操作(如offer方法)。

  • 无界扩容PriorityBlockingQueue 是基于数组实现的,当队列中的元素数量超过当前数组的容量时,它会自动进行扩容操作,以确保能够继续存储新的元素。

扩容问题

PriorityBlockingQueue 并不是一个动态调整容量的队列。它的“无界”特性并不是指它会动态地调整其内部存储的大小,而是指它不会因为容量限制而阻止新的元素被加入队列。相反,它会尽可能多地存储元素,直到耗尽系统的可用内存。

尽管 PriorityBlockingQueue 没有显式的容量限制,但在实际应用中,仍然需要考虑以下几点:

  1. 内存限制:虽然 PriorityBlockingQueue 没有固定的容量限制,但它仍然受到 JVM 的堆内存限制。如果队列中的元素过多,会导致内存不足,最终可能导致 OutOfMemoryError

  2. 垃圾回收:当队列中的元素被消费后,如果没有其他引用指向这些元素,垃圾回收器会自动回收这些对象所占用的内存。因此,即使队列曾经存储了大量的元素,只要这些元素被消费并且没有其他引用,内存就会被释放。

应用场景

PriorityBlockingQueue 适用于多种场景,包括但不限于:

  • 任务调度:在任务调度系统中,可以使用 PriorityBlockingQueue 来管理待执行的任务,工作线程可以从中取出优先级最高的任务来执行。

  • 资源管理:在资源有限的情况下,可以使用 PriorityBlockingQueue 来确定哪些请求或任务应该优先获得资源。

  • 并发访问:在多线程环境中,PriorityBlockingQueue 提供了安全的并发访问机制,多个线程可以同时向队列中添加或检索元素。

构造方法
  1. 创建一个初始容量为 11(默认容量)PriorityBlockingQueue 对象。
PriorityBlockingQueue()
  1. 创建一个初始容量为指定容量的 PriorityBlockingQueue 对象。
PriorityBlockingQueue(int capacity)
  1. 创建一个初始元素集合为指定集合的 PriorityBlockingQueue 对象。
PriorityBlockingQueue(Collection<? extends E> collection)
  1. 创建一个初始容量为指定容量、元素按照指定比较器优先级排序的 PriorityBlockingQueue 对象。
PriorityBlockingQueue(int c, Comparator<? super E> comparator)
使用示例

下面是一个使用 PriorityBlockingQueue 的简单示例,展示如何实现一个基于优先级的任务队列:

import java.util.concurrent.PriorityBlockingQueue;
import java.util.concurrent.TimeUnit;

class Task implements Comparable<Task> {
    private final int priority;
    private final String description;

    public Task(int priority, String description) {
        this.priority = priority;
        this.description = description;
    }

    public int getPriority() {
        return priority;
    }

    public String getDescription() {
        return description;
    }

    @Override
    public int compareTo(Task other) {
        // 优先级数值越小,优先级越高
        return Integer.compare(this.priority, other.priority);
    }

    @Override
    public String toString() {
        return "Task{" +
                "priority=" + priority +
                ", description='" + description + '\'' +
                '}';
    }
}

public class PriorityBlockingQueueExample {

    public static void main(String[] args) {
        // 创建一个 PriorityBlockingQueue 实例
        PriorityBlockingQueue<Task> taskQueue = new PriorityBlockingQueue<>();

        // 创建生产者线程
        Thread producer = new Thread(() -> {
            try {
                // 生成不同优先级的任务
                taskQueue.put(new Task(3, "Medium Priority Task"));
                taskQueue.put(new Task(1, "High Priority Task"));
                taskQueue.put(new Task(5, "Low Priority Task"));
                taskQueue.put(new Task(2, "Normal Priority Task"));
                System.out.println("所有任务已添加到队列中");
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt(); // 设置线程中断状态
                System.err.println("生产者线程被中断");
            }
        });

        // 创建消费者线程
        Thread consumer = new Thread(() -> {
            while (true) {
                try {
                    Task task = taskQueue.take();
                    System.out.println("处理任务: " + task);
                    TimeUnit.MILLISECONDS.sleep(1000); // 模拟任务处理时间
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt(); // 设置线程中断状态
                    System.err.println("消费者线程被中断");
                    break; // 终止循环
                }
            }
        });

        // 启动生产者线程和消费者线程
        producer.start();
        consumer.start();

        // 等待生产者线程结束
        try {
            producer.join();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt(); // 设置线程中断状态
            System.err.println("主线程被中断");
        }

        // 中断消费者线程
        consumer.interrupt();
    }
}

示例说明:

  • 优先级:在这个例子中,优先级数值越小,优先级越高。因此,优先级为 1 的任务会被首先处理。
  • 队列无界PriorityBlockingQueue 默认是无界的,即没有固定的容量限制。但是,如果队列中的元素数量过大,可能会导致内存溢出。
  • 线程中断:在捕获 InterruptedException 异常时,需要调用 Thread.currentThread().interrupt() 来恢复线程的中断状态,以便其他可能依赖中断状态的部分能够正确响应。
  • 消费者线程的终止:在生产者线程结束后,通过中断消费者线程来终止其无限循环。这是为了防止消费者线程无限期地运行下去。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2107593.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

「C++」类和对象(2)

欢迎大家来到小鸥的类和对象第二篇博客~ 目录 类的默认成员函数 构造函数 构造函数的特点&#xff1a; 析构函数 析构函数的特点&#xff1a; 拷贝构造函数 拷贝构造的特点&#xff1a; 结语&#xff1a; 本篇会着重讲解类和对象中的难点&#xff1a;类的默认成员函数 …

Ubuntu环境的MySql下载安装

下载压缩包 此文章下载的mysql版本位5.7.29 sudo wget https://downloads.mysql.com/archives/get/p/23/file/mysql-server_5.7.29-1ubuntu18.04_amd64.deb-bundle.tar解压缩 sudo tar -xvf mysql-server_5.7.29-1ubuntu18.04_amd64.deb-bundle.tar命令解释 -x&#xff1a;…

鸿蒙Next-拉起支付宝的三种方式——教程

鸿蒙Next-拉起支付宝的三种方式——教程 鸿蒙Next系统即将上线&#xff0c;应用市场逐渐丰富、很多APP都准备接入支付宝做支付功能&#xff0c;目前来说有三种方式拉起支付宝&#xff1a;通过支付宝SDK拉起、使用OpenLink拉起、传入支付宝包名使用startAbility拉起。以上的三种…

走心机做不锈钢哪个牌子好

不锈钢是现代生活中不可或缺的材料&#xff0c;它广泛应用于厨房用具、家具、建筑等领域。在市场上&#xff0c;有许多不锈钢需要加工零件供消费者选择&#xff0c;那么在选择不锈钢加工零件时制品时&#xff0c;应该如何选择数控走心机&#xff0c;找到最好的品牌呢&#xff1…

CodeSys中动态切换3D模型

文章目录 需求研究结果 需求 在前面的【CodeSys开发3d机械臂显示控件】中&#xff0c;我们已经实现了一个可以显示3d模型的控件。但是这个控件是和使用的3d模型绑定死的&#xff0c;在安装这个控件时就已经将模型文件于控件一起安装到codesys中。 假如我想在不同的工程中&…

Numpy中type()、ndim、shape、size、dtype、astype的用法

目录 numpy基础介绍示例分析及总结&#xff1a;itemsize、nbytes函数 numpy基础介绍 Numpy 补充了Python语言所欠缺的数值计算能力,是其它数据分析及机器学习库的底层库。因其完全标准C语言实现&#xff0c;运行效率充分优化。最重要一点是开源免费。numpy的核心是矩阵&#x…

思维导图怎么画好看又简单?5个软件帮助你快速进行思维导图绘制

思维导图怎么画好看又简单&#xff1f;5个软件帮助你快速进行思维导图绘制 思维导图是一种有效的思维整理和展示工具&#xff0c;可以帮助你将复杂的想法进行可视化&#xff0c;提升工作和学习效率。为了让思维导图既好看又简单&#xff0c;选择合适的软件能够大大提高绘制效率…

UE中Camera Clip截面修改

UE中Camera无法修改远截面&#xff08;FarClipingPlane)&#xff0c;只可修改近截面&#xff08;NearClipingPlane&#xff09;&#xff1a; 至于为什么无法修改远截面&#xff0c;看下代码&#xff0c;尝试继承UE的CameraComponent打印出相机投影矩阵&#xff1a; #include …

python编程知识(实现数据加密和解密)

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;开发者-曼亿点 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 曼亿点 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a…

在模板中使用 Django 会话

在 Django 中使用会话&#xff08;session&#xff09;可以让你在用户访问网站的过程中存储和访问临时数据。我们可以利用会话在速度计算器的例子中存储和显示上次计算的结果。 1、问题背景 在 Django 中&#xff0c;可以使用会话来存储用户数据。在某些情况下&#xff0c;我们…

双绞线如何抑制传导干扰

一&#xff0e;案例简介 产品去做CE认证时&#xff0c;被告知传导抗扰未通过&#xff0c;网络会断连。 剖开网线外皮&#xff0c;发现内部是散装的&#xff0c;非双绞线。因此换成双绞线网线&#xff0c;复测&#xff0c;传导抗扰通过了。 图1 非双绞线和双绞线示意图 为什么…

经销商数据对接方案:借助轻易云数据集成平台实现高效互联

在现在很多品牌方的实际需求中&#xff0c;品牌商与经销商之间的渠道博弈日益激烈。品牌商渴望掌握经销商的销量和库存数据&#xff0c;以便更好地规划生产和库存&#xff0c;提升品牌影响力&#xff1b;经销商则期望在避免库存积压的同时抢占市场&#xff0c;加速资金周转以获…

活动预告|“AI+Security”系列第3期:AI安全智能体,重塑安全团队工作范式

由安全极客、Wisemodel社区、InForSec网络安全研究国际学术论坛和海升集团联合主办的 “AISecurity”系列第3期&#xff1a; AI 安全智能体&#xff0c;重塑安全团队工作范式 线下活动 将于2024年9月11日下午14:00 在中关村智造大街G座路演厅 正式举行 欢迎扫描海报中二…

Javaweb(前端)

目录 Web开发 Web前端 HTMLCSS 盒子模型 JavaScript js引入方式 js基础语法 js函数 js对象&#xff08;JSONBOMDOM&#xff09; js事件监听 Vue ​编辑Vue快速入门 Vue常用指令 Vue生命周期 Ajax 原生Ajax Axios YApi&#xff08;应用网页&#xff09; 前端工程…

打造一流的研发型企业--- 金发科技研发驱动力初探

2006年3月29日&#xff0c;国家发改委副主任欧新黔亲自为金发科技颁发了“中国改性塑料行业第一位”、“中国合成材料制造业十强”、“中国石油化工全行业百强”三块铜牌证书&#xff0c;金发科技终于成为名符其实的行业“老大”。公司产品销售额增长迅速&#xff0c; 2006年完…

Java健康养老智慧相伴养老护理小程序系统源码代办陪诊陪护更安心

健康养老&#xff0c;智慧相伴 —— 养老护理小程序&#xff0c;代办陪诊陪护更安心 &#x1f308;【开篇&#xff1a;智慧养老&#xff0c;新时代的温馨守护】&#x1f308; 在这个快节奏的时代&#xff0c;我们总希望能给予家人更多的关爱与陪伴&#xff0c;尤其是家中的长…

【AIGC赋能】短视频创作新纪元:一键生成,爆款不再难!

文章目录 一、AI技术的深度融入&#xff1a;从辅助到主导二、实际应用场景&#xff1a;覆盖创作全流程三、展望未来&#xff1a;AI短视频创作的无限可能 《AI短视频生成与剪辑实战108招&#xff1a;ChatGPT剪映》编辑推荐内容简介作者简介目录前言/序言内页插图 在数字化浪潮的…

IP地址是怎么实现HTTPS访问的?

首先&#xff0c;需要明确的是&#xff0c;IP地址&#xff08;Internet Protocol Address&#xff09;是互联网上设备&#xff08;如服务器、路由器等&#xff09;的唯一标识符&#xff0c;它允许数据包在网络中正确地路由和传输。然而&#xff0c;IP地址本身并不直接支持HTTPS…

cesium可不可以改变影像底图颜色,如何给地球底图影像添加一层滤镜蒙版?

废话&#xff1a;你的球是不是很丑&#xff1f;是不是没有科技感&#xff1f;是不是没有好看的影像&#xff1f; 因果&#xff1a; 因&#xff1a;客户问&#xff0c;底图可不可以改变颜色&#xff0c;想让球更漂亮一些。 答&#xff1a;可以改变影像饱和度&#xff0c;透明度…

【MATLAB源码-第164期】基于matlab的轴承故障三种谱图:细化谱,功率谱,倒谱对比分析仿真。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 轴承故障分析是一种重要的维护和监控手段&#xff0c;能够帮助工程师及时发现和解决轴承在运行中可能遇到的各种问题。在轴承故障诊断中&#xff0c;通常会使用到三种谱图分析方法&#xff1a;细化谱&#xff08;Fine Spectr…