Java阻塞队列:ArrayBlockingQueue

Java阻塞队列:ArrayBlockingQueue

ArrayBlockingQueue是Java中的一个阻塞队列(Blocking Queue)实现,它是线程安全的,并且基于数组实现。ArrayBlockingQueue常用于生产者-消费者模型,在这种模型中,生产者线程负责将元素放入队列,而消费者线程负责从队列中取出元素。

ArrayBlockingQueue是一个有界队列,这意味着它有一个固定的容量。在队列已满时,试图向队列中添加元素的操作将被阻塞,直到队列有空间可用。同样地,在队列为空时,试图从队列中取出元素的操作也将被阻塞,直到队列中有可用的元素。

主要特性

  • 线程安全ArrayBlockingQueue内部使用锁和条件变量来确保线程安全。
  • 有界:队列的容量在创建时指定,并且无法改变。
  • FIFO顺序:元素按照先进先出的顺序进行处理。

构造方法

ArrayBlockingQueue提供了多个构造方法,常用的有以下两种:

public ArrayBlockingQueue(int capacity)
public ArrayBlockingQueue(int capacity, boolean fair)
  • capacity:指定队列的容量。
  • fair:指定是否使用公平策略。如果设置为true,则队列的操作将按照公平的顺序进行;否则,不保证公平性。

主要方法

  • put(E e):将指定元素添加到队列中,如果队列已满,则等待空间可用。
  • take():从队列中获取并移除元素,如果队列为空,则等待元素可用。
  • offer(E e):尝试将指定元素添加到队列中,如果队列已满,则返回false
  • poll():从队列中获取并移除元素,如果队列为空,则返回null

使用场景

ArrayBlockingQueue非常适合以下场景:

  • 生产者-消费者模型:多个生产者线程向队列中添加任务,多个消费者线程从队列中取出任务进行处理。
  • 线程池:用于存放待处理任务的队列,线程池中的工作线程从队列中取出任务并执行。

示例代码

下面是一个简单的示例,展示了如何使用ArrayBlockingQueue实现生产者-消费者模型。

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;public class ProducerConsumerExample {private static final int CAPACITY = 10;private static final BlockingQueue<Integer> queue = new ArrayBlockingQueue<>(CAPACITY);public static void main(String[] args) {Thread producer = new Thread(new Producer());Thread consumer = new Thread(new Consumer());producer.start();consumer.start();}static class Producer implements Runnable {@Overridepublic void run() {try {for (int i = 0; i < 20; i++) {System.out.println("Produced: " + i);queue.put(i);}} catch (InterruptedException e) {Thread.currentThread().interrupt();}}}static class Consumer implements Runnable {@Overridepublic void run() {try {while (true) {int value = queue.take();System.out.println("Consumed: " + value);}} catch (InterruptedException e) {Thread.currentThread().interrupt();}}}
}

线程安全

ArrayBlockingQueue的线程安全性主要依赖于内部的锁机制和条件变量来管理并发访问。这种设计确保了多个线程可以安全地进行入队和出队操作,而不会导致数据不一致或其他并发问题。具体来说,ArrayBlockingQueue通过以下几种方式实现线程安全:

ReentrantLock

ArrayBlockingQueue使用java.util.concurrent.locks.ReentrantLock来管理对共享资源的访问。ReentrantLock是一种可重入的互斥锁,允许同一个线程多次获得锁而不会导致死锁。ArrayBlockingQueue通常会使用两种锁:

  • 主锁(Main Lock):用于保护队列的所有变更操作,如插入、删除等。
  • 分离锁(Separate Locks):在某些实现中,可能会为插入和删除操作使用不同的锁,以减少锁竞争并提高并发性能。

ArrayBlockingQueue中,通常只有一个锁来保护整个队列。

Condition条件变量

ArrayBlockingQueue还使用了java.util.concurrent.locks.Condition条件变量来实现线程间的协作。Condition变量提供了类似Object类中的waitnotifynotifyAll方法,但更强大和灵活。通过Condition变量,可以让线程在特定条件下等待或被唤醒,这对于实现阻塞操作非常重要。

ArrayBlockingQueue中,通常会有两个Condition变量:

  • notFull:表示队列未满的条件。当队列已满时,试图执行插入操作的线程会在这个条件上等待,直到有空间可用。
  • notEmpty:表示队列不为空的条件。当队列为空时,试图执行移除操作的线程会在这个条件上等待,直到有元素可用。

线程安全机制的实现

以下是ArrayBlockingQueue实现线程安全的几个关键点:

  1. 加锁与解锁

    在每次修改队列状态(如插入或删除元素)之前,ArrayBlockingQueue都会先获取主锁,以确保只有一个线程能够进行修改操作。当操作完成后,再释放锁。

    final ReentrantLock lock = this.lock;
    lock.lock();
    try {// 修改队列状态
    } finally {lock.unlock();
    }
    
  2. 等待和通知

    使用Condition变量来处理队列满和空的情况。当队列已满时,插入操作会调用notFull.await()进入等待状态,直到有空间可用。同样,当队列为空时,移除操作会调用notEmpty.await()进入等待状态,直到有新元素被插入。

    final ReentrantLock lock = this.lock;
    lock.lock();
    try {while (count == items.length) {notFull.await();}// 插入元素notEmpty.signal();
    } finally {lock.unlock();
    }
    
    final ReentrantLock lock = this.lock;
    lock.lock();
    try {while (count == 0) {notEmpty.await();}// 移除元素notFull.signal();
    } finally {lock.unlock();
    }
    

完整示例代码

以下是ArrayBlockingQueue的一个简化示例,展示了如何使用ReentrantLock和Condition来实现线程安全的阻塞队列:

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;public class SimpleArrayBlockingQueue<E> {private final E[] items;private int putIndex, takeIndex, count;private final ReentrantLock lock = new ReentrantLock();private final Condition notFull = lock.newCondition();private final Condition notEmpty = lock.newCondition();public SimpleArrayBlockingQueue(int capacity) {items = (E[]) new Object[capacity];}public void put(E e) throws InterruptedException {final ReentrantLock lock = this.lock;lock.lock();try {while (count == items.length) {notFull.await();}items[putIndex] = e;if (++putIndex == items.length) putIndex = 0;count++;notEmpty.signal();} finally {lock.unlock();}}public E take() throws InterruptedException {final ReentrantLock lock = this.lock;lock.lock();try {while (count == 0) {notEmpty.await();}E e = items[takeIndex];if (++takeIndex == items.length) takeIndex = 0;count--;notFull.signal();return e;} finally {lock.unlock();}}
}

在这里插入图片描述

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

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

相关文章

[英语单词] ellipsize,动词化后缀 -ize

openvswitch manual里的一句话&#xff1a;里面有使用ellipsize&#xff0c;但是查字典是没有这个单词&#xff0c;这就是创造出来的动词。将单词ellipsis&#xff0c;加动词化后缀&#xff0c;-ize。 Often we ellipsize arguments not important to the discussion, e.g.: &…

了解protoStuff

&#x1f47d;System.out.println(“&#x1f44b;&#x1f3fc;嗨&#xff0c;大家好&#xff0c;我是代码不会敲的小符&#xff0c;目前工作于上海某电商服务公司…”); &#x1f4da;System.out.println(“&#x1f388;如果文章中有错误的地方&#xff0c;恳请大家指正&…

一文了解IO流

主要是背八股太枯燥了&#xff0c;而且目前很少用这个知识点&#xff0c;所以总结一下&#xff0c;方便记忆。 IO流&#xff08;输入/输出流&#xff09;在Java中是一个核心概念&#xff0c;广泛用于数据的读取和写入。IO流主别用于处理所有输入和输出操作&#xff0c;无论数据…

某集团数字化转型蓝图规划项目案例(94页PPT)

案例介绍&#xff1a; 本集团数字化转型蓝图规划项目通过确定目标&#xff0c;如制定集团数字化转型的整体战略和规划&#xff0c;明确转型方向和目标。构建数字化业务体系&#xff0c;实现业务流程数字化、智能化。搭建数字化管理平台&#xff0c;提升集团内部的管理效率和决…

Pyshark——安装、解析pcap文件

1、简介 PyShark是一个用于网络数据包捕获和分析的Python库&#xff0c;基于著名的网络协议分析工具Wireshark和其背后的libpcap/tshark库。它提供了一种便捷的方式来处理网络流量&#xff0c;适用于需要进行网络监控、调试和研究的场景。以下是PyShark的一些关键特性和使用方…

C语言期末考试大纲详解

一&#xff1a; C语言的基本概念 C语言是一种通用的、过程式的计算机编程语言&#xff0c;设计提供了低级内存访问和简单、灵活的语言结构。以下是关于C语言的一些基本概念和组成部分的详细解释。 文件扩展名 .c&#xff1a;这是C语言源代码文件的扩展名。它包含了程序员编…

Noisee 和 Suno创作十二生肖震撼视频 – 有详细的实操步骤

历史文章 日赚800&#xff0c;利用淘宝/闲鱼进行AI音乐售卖实操 如何让AI生成自己喜欢的歌曲-AI音乐创作的正确方式 抖音主播/电商人员有福了&#xff0c;利用Suno创作产品宣传&#xff0c;让产品动起来-小米Su7 用sunoAI写粤语歌的方法&#xff0c;博主已经亲自实践可行 …

docker desktop for mac os如何使用本地代理

在macbook上弄了个代理&#xff0c;然后按照网上所说的去配代理 然后测试下 docker pull busybox 结果无反应&#xff0c;超时。我去&#xff01;&#xff01;&#xff01; 鼓捣了半天&#xff0c;看了docker官网&#xff0c;问了chatgpt &#xff0c;按照它们所说的试了下也没…

想学编程,什么语言最好上手?

Python是许多初学者的首选&#xff0c;因为它的语法简洁易懂&#xff0c;而且有丰富的资源和社区支持。我这里有一套编程入门教程&#xff0c;不仅包含了详细的视频 讲解&#xff0c;项目实战。如果你渴望学习编程&#xff0c;不妨点个关注&#xff0c;给个评论222&#xff0c;…

Web前端中表示上标:深度解析与实战技巧

Web前端中表示上标&#xff1a;深度解析与实战技巧 在Web前端开发中&#xff0c;表示上标是一个常见的需求&#xff0c;尤其是在处理数学公式、化学符号或者其他需要特殊格式化的文本时。本文将从四个方面、五个方面、六个方面和七个方面深入探讨如何在Web前端中表示上标&…

Gone框架介绍26 - Gone v1.x 版本 正式发布,更加强大的依赖注入,更加卓越的执行效率

gone是可以高效开发Web服务的Golang依赖注入框架 github地址&#xff1a;https://github.com/gone-io/gone 文档地址&#xff1a;https://goner.fun/zh/ 文章目录 优化和新特性gone 核心功能增强内置Goners覆盖测试 后续计划 优化和新特性 gone 核心功能增强 重构了函数参数依…

【区分】累次极限与二重极限

累次极限与二重极限不要混淆&#xff0c;区分好下面5个命题

Java Character类:功能、应用与最佳实践

Java中的Character类是一种用于处理字符数据的包装类&#xff0c;属于java.lang包。作为基础类型char的包装类&#xff0c;Character提供了一系列静态方法来操作和检索字符的各种特性。这包括字符的类别判断&#xff08;如字母、数字、空白等&#xff09;、大小写转换、字符比较…

Python学习笔记13 -- API的说明及使用

一、API说明 1、API&#xff1a;Application Programming Interface -- 接口 2、Restful API&#xff1a;基于网页的API 3、JSON&#xff1a;JavaScript objection Notation 4、很多数据都可以找到公用API 二、安装requests模块 在终端输入&#xff1a; 可以保证使用当前…

os.system() 函数

os.system() 是 Python 标准库 os 模块中的一个函数&#xff0c;用于在子终端中运行系统命令。它可以在 Python 脚本中调用外部命令或程序。具体来说&#xff0c;它通过执行命令字符串并返回执行状态来实现这一点。下面是对 os.system() 函数的详细解释&#xff1a; import os…

第66集《摄大乘论》

请大家打开《讲义》第二二二页&#xff1a; 庚九、念(分二&#xff1a;辛一正念法身&#xff1b;辛二兼显净土) 辛一、正念法身(分二&#xff1a;壬一征&#xff1b;壬二释) 壬一、征 这个是讲到十门分别(二0三页)&#xff0c;分别清净法身的第九段&#xff0c;讲到念&…

单片机多个中断源时的设计思路,(51为例)工作寄存器R0-R7

51单片机中四组工作寄存器&#xff08;R0-R7&#xff09; 参考 可以看出每个工作寄存器区有8个字节即为R0-R7&#xff0c;当不指定使用哪个工作寄存器区的时候默认0区。其他工作区作为普通的RAM使用。特殊功能寄存器中有可以位寻址和不能位寻址的区域 下面文字引用 通过修改…

常用静止轨道卫星(geostationary satellite)及相关产品

1. 数据简介 维基百科 link 地球静止轨道&#xff08;或称地球赤道同步轨道&#xff0c;英语&#xff1a;geostationary orbit&#xff0c;简写&#xff1a;GEO&#xff09;是指地球赤道面上方35,786km的圆形轨道&#xff0c;该轨道上航天器的运行方向和地球自转方向一致。在…

使用python绘制三维散点图

使用python绘制三维散点图 三维散点图三维散点图的用途效果代码 三维散点图 三维散点图&#xff08;3D Scatter Plot&#xff09;是一种用于展示三维数据的图表。与二维散点图类似&#xff0c;三维散点图通过点在三维空间中的位置来表示数据点的三个特征。每个点在 x、y 和 z …

突破数据存储瓶颈!转转业财系统亿级数据存储优化实践--图文解析

突破数据存储瓶颈&#xff01;转转业财系统亿级数据存储优化实践–图文解析 原文链接&#xff1a;https://juejin.cn/post/7358704806779437097 原文作者&#xff1a;转转技术团队 业财系统&#xff1a;业务和财务一体化系统 与传统财务记账不同&#xff0c;一笔金额不再是…