Java阻塞队列:ArrayBlockingQueue

news2024/11/24 11:41:11

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 {
        @Override
        public 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 {
        @Override
        public 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.coloradmin.cn/o/1828058.html

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

相关文章

移动硬盘打不开怎么办?原因解析!

移动硬盘是一种方便携带、快速传输大量数据的存储设备。但有时我们会遇到这样的问题&#xff1a;插上电脑后&#xff0c;移动硬盘无法打开&#xff0c;出现各种错误提示。这时候我们该怎么办呢&#xff1f; 以下是一些可能导致移动硬盘打不开的原因及解决方法&#xff1a; 1.硬…

47.PyCharm P版突然无法启动

目录 1.启动cmd.exe&#xff0c;进到pycharm\bin目录&#xff0c;启动.\pycharm.bat&#xff0c;如果正常&#xff0c;就像下面这个样子&#xff0c;如果不正常&#xff0c;则会报错&#xff0c; 2.用记事本打开pycharm.bat文件&#xff0c;加上以下代码后 今晨&#xff0c;无…

llama3-70B体验

NVIDIA LLAMA3-70B大模型体验地址&#xff1a; NVIDIA NIM | llama3-70b 问题几个关于宇宙的问题&#xff0c;答案挺有意思的&#xff0c;很有启发性&#xff0c;记录一下&#xff1a; 问题1&#xff1a;既然相对论认为时间是相对的&#xff0c;为何却说宇宙寿命有137亿年&a…

股指期货功能

其金融期货的本质&#xff0c;决定了股指期货具有以下几方面特点&#xff1a; &#xff08;1&#xff09;交割方式为现金交割&#xff1b; &#xff08;2&#xff09;股指期货的持有成本较低&#xff1b; &#xff08;3&#xff09;股指期货的保证金率较低&#xff0c;杠杆性…

【尚庭公寓SpringBoot + Vue 项目实战】图片上传(十)

【尚庭公寓SpringBoot Vue 项目实战】图片上传&#xff08;十&#xff09; 文章目录 【尚庭公寓SpringBoot Vue 项目实战】图片上传&#xff08;十&#xff09;1、图片上传流程2、图片上传接口查看3、代码开发3.1、配置Minio Client3.2、开发上传图片接口 4、异常处理 1、图片…

12.泛型、trait和生命周期(中)

标题 五、定义共享的行为5.1 定义trait5.2 实现trait5.4 将trait作为参数5.5 Trait Bound5.6 返回实现了trait的类型5.7 使用trait bounds修复get_max_value_from_vector的错误5.8 使用trait bound有条件地实现方法 五、定义共享的行为 如果告诉Rust编译器某个特定类型拥有可能…

【办公类-04-03】华为助手导出照片视频分类(根据图片、视频的文件名日期分类导出)

背景需求&#xff1a; 用华为手机助手导出的照片视频&#xff0c;只能将jpg照片&#xff08;exifread读取图片的exif拍摄日期&#xff0c;Png、JPEG、mp4都无法识别到exif信息&#xff09; 【办公类-04-02】华为助手导出照片&#xff08;jpg&#xff09;读取拍摄时间分类导出…

Linux——ansible的应用

要让ansible管理业务里的主机 1.得先知道&#xff0c;有哪些主机 用IP地址&#xff0c;用主机名 2.知道了有哪些主机以后&#xff0c;精细、细分管理 主机要用某些办法&#xff0c;分组管理 在ansible里&#xff0c;要用一个东西&#xff1a;清单->inventory inventory …

C学习自学笔记-会陆续完善对应章节编程经典例子

C学习笔记 0>C语言概述 为什么学习C语言 1&#xff09;C的起源和发展------了解即可 B语言、C语言、C语言的产生地&#xff1a;都出自 美国贝尔实验室 2&#xff09;C的特点 优点&#xff1a;代码量小、速度快、功能强大 缺点&#xff1a;危险性高、开发周期长、可移植性…

POC EXP | woodpecker插件编写

woodpecker插件编写 目录 woodpecker介绍woodpecker使用插件编写 安装环境 woodpecker-sdkwoodpecker-request 创建Maven项目 Confluence OGNL表达式注入漏洞插件编写 创建Package包和Class类编写POC 漏洞POC代码编写导出jar包将jar包放入woodpecker的plugin目录运行woodpeck…

2.2 抽头

目录 为什么要抽头 什么是抽头 接入系数 怎么抽头 信号源端抽头 负载端抽头 例题分析 要点总结 为什么要抽头 阻抗转换&#xff0c;使信号源内阻Rs与负载电阻RL变得很大&#xff0c;分流小&#xff0c;再使用并联方式。 什么是抽头 接入系数 电容越大&#xff0c;分压越…

Hadoop 2.0:主流开源云架构(四)

目录 五、Hadoop 2.0访问接口&#xff08;一&#xff09;访问接口综述&#xff08;二&#xff09;浏览器接口&#xff08;三&#xff09;命令行接口 六、Hadoop 2.0编程接口&#xff08;一&#xff09;HDFS编程&#xff08;二&#xff09;Yarn编程 五、Hadoop 2.0访问接口 &am…

最大连续子序列和问题详解

最大连续子序列和问题如下&#xff1a;给定一个数字序列&#xff0c;求i,j&#xff0c;使得最大&#xff0c;输出这个最大和。 这个问题如果用暴力来做&#xff0c;枚举左端点和右端点&#xff0c;需要的复杂度&#xff0c;而计算需要的复杂度&#xff0c;因此总时间复杂度为。…

表面声波滤波器——压电材料(2)

声表面波压电材料 压电效应&#xff1a; 1880年&#xff0c;法国物理学家居里兄弟(PCunie 和J.Curie)发现将重物置于α石英晶体上,品体部分表面会产生电荷&#xff0c;电荷量与所受压力成正比正压电效应&#xff1a;外加压力作应下在表面间产生电位差。 和逆压电效&#xff1a…

14 学习PID--步进电机梯形加减速实现原理

步进电机加减速使用的场景有那些呢&#xff1f;为什么要使用加减速呢&#xff1f; 硬件驱动细分器与软件的细分参数或定时器分频参数设置不当时启动电机时&#xff0c;会遇见步进电机有啸叫声但是不会转动&#xff0c;这是因为软件产生脉冲的频率大于步进电机的启动频率&#x…

Iptables深入浅出

1、iptables的基本概念 众所周知iptables是Linux系统下自带免费的包过滤防火墙。其实不然&#xff0c;iptables其实不是真正的防火墙&#xff0c;我们可以把它理解成一个客户端代理&#xff0c;用户通过iptables这个代理&#xff0c;将用户的安全设定执行到对应的”安全框架”…

【Oracle APEX开发小技巧1】转换类型实现显示小数点前的 0 以 及常见类型转换

在 apex 交互式式网格中&#xff0c;有一数值类型为 NUMBER&#xff0c;保留小数点后两位的项&#xff0c;在 展示时小数点前的 0 不显示。 效果如下&#xff1a; 转换前&#xff1a; m.WEIGHT_COEFFICIENT 解决方案&#xff1a; 将 NUMBER&#xff08;20&#xff0c;2&#xf…

Python进阶:从函数到文件的编程艺术!!!

第二章&#xff1a;Python进阶 模块概述 函数是一段可重复使用的代码块&#xff0c;它接受输入参数并返回一个结果。函数可以用于执行特定的任务、计算结果、修改数据等&#xff0c;使得代码更具模块化和可重用性。 模块是一组相关函数、类和变量的集合&#xff0c;它们被封…

Android Media Framework(六)插件式编程与OMXStore

OpenMAX IL Spec阅读到上一节就结束了&#xff0c;这一节开始正式进入到Framework阅读阶段&#xff0c;我们将了解OpenMAX框架是如何与Android Framework连接的。 1、插件式编程 插件式编程&#xff08;Plugin-based Programming&#xff09;是一种软件开发模式&#xff0c;它…

docker login 报错: http: server gave HTTP response to HTTPS client

环境&#xff1a; 自建 Harbor、Docker 1. 问题分析 # 命令&#xff0c;这里用的是 IP&#xff0c;可以为域名 docker login -u test 172.16.51.182:31120 # 输入密码 Password:# 报错如下&#xff1a; Error response from daemon: Get "https://172.16.51.182:31120/…