ArrayBlockingQueue 和 LinkedBlockingQueue 有什么区别?

简介: ArrayBlockingQueue 和 LinkedBlockingQueue 有什么区别?

!! 有的时候博客内容会有变动,首发博客是最新的,其他博客地址可能会未同步,认准https://blog.zysicyj.top

首发博客地址[1]

面试题手册[2]

系列文章地址[3]


1. 什么是 ArrayBlockingQueue 和 LinkedBlockingQueue?

  • ArrayBlockingQueue:是一个基于数组实现的有界阻塞队列,它按照先进先出(FIFO)的原则对元素进行排序。
  • LinkedBlockingQueue:是一个基于链表实现的可选有界或无界阻塞队列,它也按照先进先出(FIFO)的原则对元素进行排序。

2. 为什么需要 ArrayBlockingQueue 和 LinkedBlockingQueue?

在多线程编程中,我们经常需要使用队列来实现线程间的数据共享。而阻塞队列是一种特殊的队列,当队列为空时,从队列中获取元素的操作会被阻塞;当队列满时,往队列中添加元素的操作会被阻塞。这样可以有效地控制线程之间的协作和同步。

ArrayBlockingQueue 和 LinkedBlockingQueue 都是 Java 并发包提供的线程安全的阻塞队列实现,它们提供了不同的特性和适用场景。

3. ArrayBlockingQueue 和 LinkedBlockingQueue 的实现原理?

ArrayBlockingQueue

  • ArrayBlockingQueue 内部使用一个定长数组来存储元素,通过两个指针分别指向队头和队尾。
  • 当往队列中添加元素时,如果队列已满,则添加操作会被阻塞,直到有空闲位置。
  • 当从队列中获取元素时,如果队列为空,则获取操作会被阻塞,直到有可用元素。
  • ArrayBlockingQueue 使用 ReentrantLock 来保证线程安全,并通过 Condition 实现阻塞和唤醒机制。

LinkedBlockingQueue

  • LinkedBlockingQueue 内部使用一个链表来存储元素,每个节点包含一个元素和指向下一个节点的引用。
  • 当往队列中添加元素时,如果队列已满(对于有界队列),则添加操作会被阻塞,直到有空闲位置。
  • 当从队列中获取元素时,如果队列为空,则获取操作会被阻塞,直到有可用元素。
  • LinkedBlockingQueue 使用两把锁分别控制队头和队尾的访问,以提高并发性能。

4. ArrayBlockingQueue 和 LinkedBlockingQueue 的使用示例

ArrayBlockingQueue 示例:

import java.util.concurrent.ArrayBlockingQueue;
public class ArrayBlockingQueueExample {
    public static void main(String[] args) throws InterruptedException {
        // 创建一个容量为3的ArrayBlockingQueue
        ArrayBlockingQueue<Integer> queue = new ArrayBlockingQueue<>(3);
        // 往队列中添加元素
        queue.put(1);
        queue.put(2);
        queue.put(3);
        // 队列已满,添加操作会被阻塞
        queue.put(4);
        // 从队列中获取元素
        int element = queue.take();
        System.out.println("取出元素:" + element);
        // 队列已空,获取操作会被阻塞
        int element2 = queue.take();
    }
}

LinkedBlockingQueue 示例:

import java.util.concurrent.LinkedBlockingQueue;
public class LinkedBlockingQueueExample {
    public static void main(String[] args) throws InterruptedException {
        // 创建一个容量为3的LinkedBlockingQueue
        LinkedBlockingQueue<Integer> queue = new LinkedBlockingQueue<>(3);
        // 往队列中添加元素
        queue.put(1);
        queue.put(2);
        queue.put(3);
        // 队列已满,添加操作会被阻塞
        queue.put(4);
        // 从队列中获取元素
        int element = queue.take();
        System.out.println("取出元素:" + element);
        // 队列已空,获取操作会被阻塞
        int element2 = queue.take();
    }
}

5. ArrayBlockingQueue 和 LinkedBlockingQueue 的优点

  • ArrayBlockingQueue:
  • 内部使用数组实现,读写性能较高。
  • 支持有界队列,可以限制队列的大小。
  • LinkedBlockingQueue:
  • 内部使用链表实现,插入和删除性能较高。
  • 支持可选的有界或无界队列。

6. ArrayBlockingQueue 和 LinkedBlockingQueue 的缺点

  • ArrayBlockingQueue:
  • 容量固定,不支持动态扩容。
  • 在并发情况下,可能存在线程饥饿问题(某些线程一直无法获取到锁)。
  • LinkedBlockingQueue:
  • 内存消耗较大,因为每个节点都需要额外的空间来存储引用。
  • 在并发情况下,可能存在线程饥饿问题(某些线程一直无法获取到锁)。

7. ArrayBlockingQueue 和 LinkedBlockingQueue 的使用注意事项

  • ArrayBlockingQueue:
  • 需要指定队列的容量大小。
  • 当队列已满时,添加操作会被阻塞。
  • 当队列为空时,获取操作会被阻塞。
  • LinkedBlockingQueue:
  • 可以选择有界或无界队列。
  • 当队列已满时(对于有界队列),添加操作会被阻塞。
  • 当队列为空时,获取操作会被阻塞。

8. 总结

ArrayBlockingQueue 和 LinkedBlockingQueue 是 Java 并发包提供的线程安全的阻塞队列实现。它们分别基于数组和链表来存储元素,并提供了不同的特性和适用场景。ArrayBlockingQueue 适合读写性能较高、固定容量的场景;而 LinkedBlockingQueue 适合插入和删除性能较高、可选有界或无界的场景。在使用时需要根据具体需求选择合适的队列实现。

参考资料

[1]

首发博客地址: https://blog.zysicyj.top/

[2]

面试题手册: https://store.amazingmemo.com/chapterDetail/1685324709017001

[3]

系列文章地址: https://blog.zysicyj.top/categories/技术文章/后端技术/系列文章/面试题精讲/

本文由 mdnice 多平台发布

相关文章
|
存储 缓存 文件存储
如何保证分布式文件系统的数据一致性
分布式文件系统需要向上层应用提供透明的客户端缓存,从而缓解网络延时现象,更好地支持客户端性能水平扩展,同时也降低对文件服务器的访问压力。当考虑客户端缓存的时候,由于在客户端上引入了多个本地数据副本(Replica),就相应地需要提供客户端对数据访问的全局数据一致性。
32701 79
如何保证分布式文件系统的数据一致性
|
前端开发 容器
HTML5+CSS3前端入门教程---从0开始通过一个商城实例手把手教你学习PC端和移动端页面开发第8章FlexBox布局(上)
HTML5+CSS3前端入门教程---从0开始通过一个商城实例手把手教你学习PC端和移动端页面开发第8章FlexBox布局
17756 20
|
设计模式 存储 监控
设计模式(C++版)
看懂UML类图和时序图30分钟学会UML类图设计原则单一职责原则定义:单一职责原则,所谓职责是指类变化的原因。如果一个类有多于一个的动机被改变,那么这个类就具有多于一个的职责。而单一职责原则就是指一个类或者模块应该有且只有一个改变的原因。bad case:IPhone类承担了协议管理(Dial、HangUp)、数据传送(Chat)。good case:里式替换原则定义:里氏代换原则(Liskov 
36685 20
设计模式(C++版)
|
存储 编译器 C语言
抽丝剥茧C语言(初阶 下)(下)
抽丝剥茧C语言(初阶 下)
|
机器学习/深度学习 人工智能 自然语言处理
带你简单了解Chatgpt背后的秘密:大语言模型所需要条件(数据算法算力)以及其当前阶段的缺点局限性
带你简单了解Chatgpt背后的秘密:大语言模型所需要条件(数据算法算力)以及其当前阶段的缺点局限性
24765 14
|
机器学习/深度学习 弹性计算 监控
重生之---我测阿里云U1实例(通用算力型)
阿里云产品全线降价的一力作,2023年4月阿里云推出新款通用算力型ECS云服务器Universal实例,该款服务器的真实表现如何?让我先测为敬!
36665 15
重生之---我测阿里云U1实例(通用算力型)
|
SQL 存储 弹性计算
Redis性能高30%,阿里云倚天ECS性能摸底和迁移实践
Redis在倚天ECS环境下与同规格的基于 x86 的 ECS 实例相比,Redis 部署在基于 Yitian 710 的 ECS 上可获得高达 30% 的吞吐量优势。成本方面基于倚天710的G8y实例售价比G7实例低23%,总性价比提高50%;按照相同算法,相对G8a,性价比为1.4倍左右。
|
存储 算法 Java
【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的限流器RateLimiter功能服务
随着互联网的快速发展,越来越多的应用程序需要处理大量的请求。如果没有限制,这些请求可能会导致应用程序崩溃或变得不可用。因此,限流器是一种非常重要的技术,可以帮助应用程序控制请求的数量和速率,以保持稳定和可靠的运行。
29839 52

热门文章

最新文章

下一篇
开通oss服务