【面试题精讲】ArrayBlockingQueue 和 LinkedBlockingQueue 区别

简介: 【面试题精讲】ArrayBlockingQueue 和 LinkedBlockingQueue 区别

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

首发博客地址

面试题手册

系列文章地址


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适合插入和删除性能较高、可选有界或无界的场景。在使用时需要根据具体需求选择合适的队列实现。

本文由 mdnice 多平台发布

相关文章
|
5月前
|
Java
【Java集合类面试二十八】、说一说TreeSet和HashSet的区别
HashSet基于哈希表实现,无序且可以有一个null元素;TreeSet基于红黑树实现,支持排序,不允许null元素。
|
5月前
|
Java
【Java集合类面试二十三】、List和Set有什么区别?
List和Set的主要区别在于List是一个有序且允许元素重复的集合,而Set是一个无序且元素不重复的集合。
|
5月前
|
存储 Java 索引
【Java集合类面试二十四】、ArrayList和LinkedList有什么区别?
ArrayList基于动态数组实现,支持快速随机访问;LinkedList基于双向链表实现,插入和删除操作更高效,但占用更多内存。
|
4月前
|
Android开发 Kotlin
Android经典面试题之Kotlin的==和===有什么区别?
本文介绍了 Kotlin 中 `==` 和 `===` 操作符的区别:`==` 用于比较值是否相等,而 `===` 用于检查对象身份。对于基本类型,两者行为相似;对于对象引用,`==` 比较值相等性,`===` 检查引用是否指向同一实例。此外,还列举了其他常用比较操作符及其应用场景。
200 93
|
2月前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
90 14
|
1月前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
2月前
|
存储 缓存 网络协议
计算机网络常见面试题(二):浏览器中输入URL返回页面过程、HTTP协议特点,GET、POST的区别,Cookie与Session
计算机网络常见面试题(二):浏览器中输入URL返回页面过程、HTTP协议特点、状态码、报文格式,GET、POST的区别,DNS的解析过程、数字证书、Cookie与Session,对称加密和非对称加密
|
3月前
|
编译器
经典面试题:变量的声明和定义有什么区别
在编程领域,变量的“声明”与“定义”是经典面试题之一。声明告诉编译器一个变量的存在,但不分配内存,通常包含变量类型和名称;而定义则为变量分配内存空间,一个变量必须至少被定义一次。简而言之,声明是告知变量形式,定义则是实际创建变量并准备使用。
|
3月前
|
XML 前端开发 Java
Spring,SpringBoot和SpringMVC的关系以及区别 —— 超准确,可当面试题!!!也可供零基础学习
本文阐述了Spring、Spring Boot和Spring MVC的关系与区别,指出Spring是一个轻量级、一站式、模块化的应用程序开发框架,Spring MVC是Spring的一个子框架,专注于Web应用和网络接口开发,而Spring Boot则是对Spring的封装,用于简化Spring应用的开发。
245 0
Spring,SpringBoot和SpringMVC的关系以及区别 —— 超准确,可当面试题!!!也可供零基础学习
|
3月前
|
前端开发 小程序 JavaScript
面试官:px、em、rem、vw、rpx 之间有什么区别?
面试官:px、em、rem、vw、rpx 之间有什么区别?
76 0