热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java数据结构与算法分析|队列

GitHub源码分享项目主页:https:github.comgozhuyinglongblog-demos本文源码:https:github.comg

GitHub源码分享


项目主页:https://github.com/gozhuyinglong/blog-demos
本文源码:https://github.com/gozhuyinglong/blog-demos/tree/main/java-data-structures



1. 队列(queue)

队列和栈一样,也是一个操作受限制的线性表。不同的是队列的插入在一端进行,我们称为队尾(rear);而删除(取出)在另一端进行,我们称为队头(front)。

队列是一个先进先出(FIFO - First In First Out)的有序列表,其操作只有两种:


  • 入队(enqueue):向队尾添加一个元素
  • 出队(dequeue):从队头删除(取出)一个元素

队列的模型


2. 队列的数组实现

如同栈一样,对队列的每一种操作,链表实现或数组实现都给出快速的运行时间。队列的链表实现是简单而直接的,我们就不过介绍了。下面我们讨论如何使用数组实现一个队列。

先看下图,我们需要声明一个数组,并维护两个指针:


  • head指针:指向待出列的元素位置
  • tail指针:指向待入列的存储位置
    可以看出,到达队尾后无法再添加新的元素,然后前面已出列的位置还空着。

数组实现队列

上面问题我们可以将数组进行首尾想死,形成一个环形数组,即指针到达数组尾部后,重新指向数组头部,如下图所示。

环形数组实现队列

这里需要注意几点:


  • 判断空队列可以通过head == tail
  • 判断队列满不能再通过head与tail重合,而是需要空出一个存储空间来,即:head == tail + 1。而环形数组需要取模运算,所以正确判断队列满:head == (tail + 1) % capacity。
  • 数组真实容量应为:指定容量 + 1

3. 代码实现

下面代码是使用环形数组实现的队列,所以又叫做环形队列
其容量为:指定容量 + 1,head 与t ail 初始值为0。队列存储元素使用了泛型,所以可以操作你想用的数据类型。下面看具体实现:

public class ArrayQueueDemo {public static void main(String[] args) {ArrayQueue<Integer> queue &#61; new ArrayQueue<>(5);System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("入队&#xff1a;1 --> " &#43; queue.add(1));System.out.println("入队&#xff1a;2 --> " &#43; queue.add(2));System.out.println("入队&#xff1a;3 --> " &#43; queue.add(3));System.out.println("入队&#xff1a;4 --> " &#43; queue.add(4));System.out.println("入队&#xff1a;5 --> " &#43; queue.add(5));System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("入队&#xff1a;6 --> " &#43; queue.add(6));System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("入队&#xff1a;7 --> " &#43; queue.add(7));System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("入队&#xff1a;8 --> " &#43; queue.add(8));System.out.println("入队&#xff1a;9 --> " &#43; queue.add(9));System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("入队&#xff1a;10 --> " &#43; queue.add(10));System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);}private static class ArrayQueue<T> {private final T[] queue; // 存储队列数据元素private final int capacity; // 容量private int head &#61; 0; // 头部指针&#xff0c;指向队头元素private int tail &#61; 0; // 尾部指针&#xff0c;指向下一个待入队元素的存储位置public ArrayQueue(int capacity) {this.capacity &#61; capacity &#43; 1; // 环形队列需要空出一个位置&#xff0c;来满足队列满时head与tail不重合this.queue &#61; (T[]) new Object[this.capacity];}/*** 向队列添加一个元素** &#64;param data* &#64;return*/public boolean add(T data) {// 队列满&#xff0c;添加失败if (isFull()) {return false;}// tail指向下一个待入队元素的存储位置&#xff0c;所以先赋值再让指针加1queue[tail] &#61; data;// 环形数组需要取模运算tail &#61; (tail &#43; 1) % capacity;return true;}/*** 从队列中获取一个元素** &#64;return*/public T get() {if (isEmpty()) {return null;}// head指向头元素位置&#xff0c;所以先取出再让指针加1T data &#61; queue[head];// 环形数组需要取模运算head &#61; (head &#43; 1) % capacity;return data;}/*** 当前队列大小** &#64;return*/public int size() {int size &#61; tail - head;if (size < 0) {size &#43;&#61; capacity;}return size;}/*** 队列是否为空&#xff1a;当tail与head指向同一位置时&#xff0c;表示队列为空** &#64;return*/public boolean isEmpty() {return tail &#61;&#61; head;}/*** 队列是否已满&#xff1a;因为预留了一个位置&#xff0c;所以tail需要加1&#xff1b;环形队列需要取模运算** &#64;return*/public boolean isFull() {return head &#61;&#61; (tail &#43; 1) % capacity;}}
}

输出结果&#xff1a;

头指针: 0 尾指针: 0 队列大小: 0 容量: 6
出队&#xff1a; --> null
入队&#xff1a;1 --> true
入队&#xff1a;2 --> true
入队&#xff1a;3 --> true
入队&#xff1a;4 --> true
入队&#xff1a;5 --> true
头指针: 0 尾指针: 5 队列大小: 5 容量: 6
出队&#xff1a; --> 1
入队&#xff1a;6 --> true
头指针: 1 尾指针: 0 队列大小: 5 容量: 6
入队&#xff1a;7 --> false
出队&#xff1a; --> 2
出队&#xff1a; --> 3
头指针: 3 尾指针: 0 队列大小: 3 容量: 6
入队&#xff1a;8 --> true
入队&#xff1a;9 --> true
头指针: 3 尾指针: 2 队列大小: 5 容量: 6
出队&#xff1a; --> 4
出队&#xff1a; --> 5
出队&#xff1a; --> 6
出队&#xff1a; --> 8
出队&#xff1a; --> 9
头指针: 2 尾指针: 2 队列大小: 0 容量: 6
入队&#xff1a;10 --> true
头指针: 2 尾指针: 3 队列大小: 1 容量: 6

推荐阅读
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • 本文介绍了解决二叉树层序创建问题的方法。通过使用队列结构体和二叉树结构体,实现了入队和出队操作,并提供了判断队列是否为空的函数。详细介绍了解决该问题的步骤和流程。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • 李逍遥寻找仙药的迷阵之旅
    本文讲述了少年李逍遥为了救治婶婶的病情,前往仙灵岛寻找仙药的故事。他需要穿越一个由M×N个方格组成的迷阵,有些方格内有怪物,有些方格是安全的。李逍遥需要避开有怪物的方格,并经过最少的方格,找到仙药。在寻找的过程中,他还会遇到神秘人物。本文提供了一个迷阵样例及李逍遥找到仙药的路线。 ... [详细]
  • JDK源码学习之HashTable(附带面试题)的学习笔记
    本文介绍了JDK源码学习之HashTable(附带面试题)的学习笔记,包括HashTable的定义、数据类型、与HashMap的关系和区别。文章提供了干货,并附带了其他相关主题的学习笔记。 ... [详细]
  • 本文介绍了使用哈夫曼树实现文件压缩和解压的方法。首先对数据结构课程设计中的代码进行了分析,包括使用时间调用、常量定义和统计文件中各个字符时相关的结构体。然后讨论了哈夫曼树的实现原理和算法。最后介绍了文件压缩和解压的具体步骤,包括字符统计、构建哈夫曼树、生成编码表、编码和解码过程。通过实例演示了文件压缩和解压的效果。本文的内容对于理解哈夫曼树的实现原理和应用具有一定的参考价值。 ... [详细]
  • HashMap的相关问题及其底层数据结构和操作流程
    本文介绍了关于HashMap的相关问题,包括其底层数据结构、JDK1.7和JDK1.8的差异、红黑树的使用、扩容和树化的条件、退化为链表的情况、索引的计算方法、hashcode和hash()方法的作用、数组容量的选择、Put方法的流程以及并发问题下的操作。文章还提到了扩容死链和数据错乱的问题,并探讨了key的设计要求。对于对Java面试中的HashMap问题感兴趣的读者,本文将为您提供一些有用的技术和经验。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
author-avatar
温思家羽绒家纺旗舰店
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有