0

0

java代码怎样实现队列及入队出队操作 java代码队列数据结构的实用实现方法​

星夢妙者

星夢妙者

发布时间:2025-08-11 18:17:02

|

313人浏览过

|

来源于php中文网

原创

最直接且推荐的方式是使用java.util.queue接口的实现类如linkedlist或arraydeque,1. 入队操作应优先使用offer()方法,因其在队列满时返回false而非抛出异常;2. 出队操作应优先使用poll()方法,因其在队列为空时返回null而非抛出异常;3. 查看头部元素应使用peek()方法以避免移除元素;4. 使用queue接口而非直接操作list能更好表达fifo意图并避免误用;5. linkedlist基于双向链表,适合频繁动态增删的场景,但内存开销大;6. arraydeque基于环形数组,性能更优、内存效率高,是多数场景下的首选;7. 在多线程环境下,应使用java.util.concurrent包中的线程安全队列,如concurrentlinkedqueue(非阻塞、高吞吐)、linkedblockingqueue(可阻塞、支持有界)或arrayblockingqueue(固定容量、基于数组);8. 应根据是否需要阻塞、容量限制和性能需求选择合适的并发队列,避免手动同步非线程安全的队列实现,以确保正确性和性能。

java代码怎样实现队列及入队出队操作 java代码队列数据结构的实用实现方法​

Java中实现队列及其入队出队操作,最直接且推荐的方式是利用

java.util.Queue
接口及其具体的实现类,比如
LinkedList
ArrayDeque
。这些类提供了符合队列先进先出(FIFO)原则的标准方法,让我们可以方便地管理数据的流入与流出。

解决方案

在Java中,队列(Queue)是一种重要的数据结构,它遵循“先进先出”(FIFO, First-In-First-Out)的原则。这意味着第一个进入队列的元素也将是第一个离开队列的。Java集合框架提供了

java.util.Queue
接口,以及多种实现类来满足不同的需求。

要实现队列及其入队(enqueue)和出队(dequeue)操作,我们通常会选择

LinkedList
ArrayDeque
。它们都实现了
Queue
接口,并提供了以下核心方法:

立即学习Java免费学习笔记(深入)”;

  • 入队操作(Enqueue)

    • offer(E e)
      :将指定元素插入队列的尾部。如果队列已满,此方法会返回
      false
      ,而不会抛出异常。这是推荐的入队方法。
    • add(E e)
      :与
      offer()
      类似,但如果队列已满(对于有容量限制的队列),它会抛出
      IllegalStateException
  • 出队操作(Dequeue)

    • poll()
      :获取并移除队列的头部元素。如果队列为空,此方法会返回
      null
      。这是推荐的出队方法。
    • remove()
      :与
      poll()
      类似,但如果队列为空,它会抛出
      NoSuchElementException
  • 查看头部元素(Peek)

    • peek()
      :获取但不移除队列的头部元素。如果队列为空,此方法会返回
      null
    • element()
      :与
      peek()
      类似,但如果队列为空,它会抛出
      NoSuchElementException

以下是一个使用

LinkedList
作为队列实现的简单示例:

import java.util.LinkedList;
import java.util.Queue;

public class SimpleQueueExample {
    public static void main(String[] args) {
        // 声明一个Queue接口类型的变量,使用LinkedList实现
        Queue messageQueue = new LinkedList<>();

        System.out.println("队列是否为空? " + messageQueue.isEmpty()); // true

        // 入队操作:使用 offer()
        messageQueue.offer("消息A");
        messageQueue.offer("消息B");
        messageQueue.offer("消息C");
        System.out.println("入队后队列: " + messageQueue); // [消息A, 消息B, 消息C]

        // 查看头部元素:使用 peek()
        String headMessage = messageQueue.peek();
        System.out.println("队列头部元素 (不移除): " + headMessage); // 消息A
        System.out.println("查看后队列: " + messageQueue); // [消息A, 消息B, 消息C]

        // 出队操作:使用 poll()
        String dequeuedMessage1 = messageQueue.poll();
        System.out.println("出队元素1: " + dequeuedMessage1); // 消息A
        System.out.println("出队后队列: " + messageQueue); // [消息B, 消息C]

        String dequeuedMessage2 = messageQueue.poll();
        System.out.println("出队元素2: " + dequeuedMessage2); // 消息B
        System.out.println("出队后队列: " + messageQueue); // [消息C]

        // 尝试从空队列出队
        messageQueue.poll(); // 移除消息C
        String emptyPollResult = messageQueue.poll();
        System.out.println("从空队列出队结果: " + emptyPollResult); // null
        System.out.println("最终队列: " + messageQueue); // []

        System.out.println("队列是否为空? " + messageQueue.isEmpty()); // true
    }
}

选择

offer()
/
poll()
/
peek()
而非
add()
/
remove()
/
element()
是更健壮的做法,尤其是在处理可能达到容量限制或可能为空的队列时,它们通过返回值而非抛出异常来指示操作结果,这在很多场景下更易于错误处理。

为什么在Java中我们更倾向于使用
Queue
接口而不是直接操作
List
实现队列?

这其实是个很好的问题,尤其对于初学者来说,可能会觉得

LinkedList
本身就是个
List
,为什么不直接用它的
add()
remove(0)
来模拟队列呢?在我看来,这主要关乎“意图表达”和“契约保证”。

当你声明一个变量为

Queue myQueue = new LinkedList<>();
时,你向所有阅读这段代码的人明确地表达了:这个集合的目的是作为一个队列来使用,它将遵循FIFO原则。这种明确性非常重要。如果我看到一个
List myList = new LinkedList<>();
,我不会立刻知道它是不是在扮演队列的角色,它可能被用于任何
List
的操作,比如随机访问
get(index)
,或者在中间插入元素
add(index, element)
,而这些操作在队列的语境下通常是不被允许或不推荐的。

Queue
接口的方法(
offer
,
poll
,
peek
)是专门为队列行为设计的,它们有清晰的语义和预期的行为,例如
poll()
在队列为空时返回
null
而不是抛出异常,这使得错误处理更加优雅。直接操作
List
的方法,比如
remove(0)
,在列表为空时会抛出
IndexOutOfBoundsException
,这在处理队列时可能需要额外的
try-catch
块,显得不够自然。

Toolplay
Toolplay

一站式AI应用聚合生成平台

下载

所以,使用

Queue
接口不仅提升了代码的可读性和可维护性,它还通过接口的约束,强制开发者以队列的方式来使用这个数据结构,避免了潜在的误用。这就像你买了一个专门用来烧水的电水壶,而不是用一个普通的锅在炉子上烧水——两者都能烧水,但电水壶的设计更符合烧水的特定场景,用起来也更安全、更方便。

LinkedList
ArrayDeque
作为队列实现,它们各自的适用场景与性能考量是什么?

在Java中,

LinkedList
ArrayDeque
是实现
Queue
接口最常用的两个类,但它们底层的数据结构和性能特性却大相径庭,因此在选择时需要根据具体场景来权衡。

LinkedList

  • 底层实现: 双向链表。每个元素都包含指向前后元素的引用。
  • 优点:
    • 动态性强: 插入和删除操作(无论在头部、尾部还是中间)的平均时间复杂度都是O(1),因为只需要修改少数几个节点的引用。这意味着它在频繁进行入队出队操作时表现稳定。
    • 内存使用灵活: 不需要预先分配固定大小的内存空间,可以根据需要动态增长。
    • 可作双端队列(Deque):
      LinkedList
      同时实现了
      Deque
      接口,这意味着它也可以作为栈(Stack)来使用,支持在两端进行高效的添加和移除。
  • 缺点:
    • 内存开销大: 每个元素除了存储数据本身,还需要额外的内存来存储前后节点的引用,导致内存效率相对较低。
    • 缓存不友好: 链表中的元素在内存中不一定是连续存储的,这可能导致CPU缓存命中率低,从而影响性能。
    • 随机访问慢: 查找特定位置的元素需要从头或尾遍历,时间复杂度为O(n)。
  • 适用场景:
    • 当队列的容量需要频繁地、大幅度地变化,且对内存的连续性要求不高时。
    • 当你需要一个既能作为队列也能作为栈,或者需要支持双端操作的数据结构时。
    • 对随机访问性能不敏感的场景。

ArrayDeque

  • 底层实现: 动态可调整大小的数组。它是一个环形数组(circular array),通过两个指针(头指针和尾指针)来管理元素的添加和移除。
  • 优点:
    • 性能优异: 对于入队和出队操作,其时间复杂度通常为O(1)(摊还常数时间),因为它利用了数组的连续性,缓存命中率高。在大多数情况下,它比
      LinkedList
      更快。
    • 内存效率高: 不需要额外的节点引用开销,存储效率更高。
    • 可作双端队列(Deque):
      ArrayDeque
      也实现了
      Deque
      接口,同样可以高效地作为栈或双端队列使用。
  • 缺点:
    • 扩容开销: 当内部数组空间不足时,需要进行扩容操作(创建一个更大的新数组并将旧数组的元素复制过去),这可能导致单次操作的开销较大(尽管摊还下来是O(1))。
  • 适用场景:
    • 绝大多数的队列和栈场景: 尤其是在对性能有较高要求,且队列大小变化相对平稳(不会频繁触发扩容)时。
    • 作为普通队列或栈使用时,
      ArrayDeque
      通常是首选,因为它在性能上往往优于
      LinkedList

我的选择倾向: 说实话,对于大多数“纯粹”的FIFO队列应用,我个人会优先考虑

ArrayDeque
。它的性能优势在实际项目中往往非常明显,尤其是在处理大量数据时。只有当我知道我可能需要链表的某些特定优势(比如在队列中间进行插入删除,或者对内存碎片有特别的考量,尽管这在队列场景下不常见)时,我才会考虑
LinkedList
。当然,如果只是一个很小的队列,性能差异可能微乎其微,那么选择哪个都无伤大雅。

在多线程环境下,Java队列的实现有哪些特殊考虑和推荐实践?

在多线程环境下使用队列,情况会变得复杂得多。标准的

LinkedList
ArrayDeque
都不是线程安全的,这意味着如果多个线程同时对它们进行入队或出队操作,很可能会导致数据损坏、丢失,或者出现意想不到的错误(比如
ConcurrentModificationException
)。我曾经就掉过这样的坑,调试起来简直是噩梦。

因此,在多线程编程中,我们必须使用专门设计用于并发访问的队列实现。Java的

java.util.concurrent
包为我们提供了强大的工具

  1. ConcurrentLinkedQueue
    :非阻塞队列

    • 特点: 这是一个基于链表的、线程安全的队列。它的特点是“非阻塞”,意味着当一个线程尝试入队或出队时,如果操作无法立即完成(例如队列为空),它不会阻塞该线程,而是通过CAS(Compare-And-Swap)操作等无锁算法来保证线程安全。
    • 优点: 高并发性能,特别适合生产者-消费者模型中,生产者和消费者数量都很多,且不希望线程因为队列满或空而长时间等待的场景。它内部的实现非常精巧,避免了锁的开销。
    • 缺点: 不支持有界队列(即容量无限)。
    • 适用场景: 高吞吐量的日志系统、事件分发系统等。
  2. LinkedBlockingQueue
    :有界/无界阻塞队列

    • 特点: 这是一个基于链表的、线程安全的阻塞队列。它支持可选的容量限制。当队列满时,尝试入队的线程会被阻塞,直到队列有空间;当队列空时,尝试出队的线程会被阻塞,直到队列有元素。
    • 优点: 提供了生产者-消费者模式中常见的流量控制机制。你可以通过构造函数指定其容量,从而防止内存溢出。
    • 缺点: 线程会因为队列状态而阻塞,在高并发下可能引入额外的上下文切换开销。
    • 适用场景: 大多数经典的生产者-消费者模型,例如消息队列、任务调度器等,需要对生产者或消费者进行流量控制的场景。
  3. ArrayBlockingQueue
    :有界阻塞队列

    • 特点: 这是一个基于数组的、线程安全的阻塞队列。它必须在创建时指定容量,且容量不可变。
    • 优点: 内部使用数组实现,具有更好的局部性,在某些情况下可能比
      LinkedBlockingQueue
      有更好的性能表现(但通常差异不大)。同样提供了阻塞机制和容量控制。
    • 缺点: 容量固定,一旦创建无法改变。
    • 适用场景: 当你需要一个固定容量的队列,且对性能有较高要求时。例如线程池中的任务队列。

推荐实践:

  • 明确需求: 首先要搞清楚你的队列是需要阻塞行为(即生产者在队列满时等待,消费者在队列空时等待)还是非阻塞行为。
  • 容量考量: 如果你需要限制队列的最大容量以防止资源耗尽,那么
    LinkedBlockingQueue
    ArrayBlockingQueue
    是你的选择。
  • 性能权衡: 对于大多数情况,
    LinkedBlockingQueue
    是一个非常通用且性能不错的选择。如果你追求极致的非阻塞性能,且不介意队列无界,可以考虑
    ConcurrentLinkedQueue
  • 避免手动同步: 永远不要尝试自己用
    synchronized
    ReentrantLock
    去包装
    LinkedList
    ArrayDeque
    来实现线程安全队列。这不仅容易出错,而且性能往往不如
    java.util.concurrent
    包中经过精心设计和优化的实现。
  • 选择最合适的: 没有“万能”的线程安全队列。理解它们各自的特点,根据你的具体应用场景(如吞吐量要求、是否需要容量限制、是否允许阻塞等)来选择最合适的实现。

在我的经验中,当涉及到多线程协作时,直接使用

java.util.concurrent
包提供的队列是唯一可靠且高效的做法。它们不仅保证了数据一致性,还考虑了各种并发场景下的性能优化,省去了我们大量重复造轮子和调试并发问题的精力。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

463

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

458

2024.03.01

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

538

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

27

2026.01.06

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1127

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

193

2025.10.17

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

391

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 3.1万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 3万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号