热门标签 | 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

推荐阅读
  • JUC(三):深入解析AQS
    本文详细介绍了Java并发工具包中的核心类AQS(AbstractQueuedSynchronizer),包括其基本概念、数据结构、源码分析及核心方法的实现。 ... [详细]
  • 在多线程并发环境中,普通变量的操作往往是线程不安全的。本文通过一个简单的例子,展示了如何使用 AtomicInteger 类及其核心的 CAS 无锁算法来保证线程安全。 ... [详细]
  • 数据结构第三章,栈、队列、数组,期末不挂科指南,第3篇
    数据结构第三章,栈、队列、数组,期末不挂科指南,第3篇,Go语言社区,Golang程序员人脉社 ... [详细]
  • 浅析python实现布隆过滤器及Redis中的缓存穿透原理_python
    本文带你了解了位图的实现,布隆过滤器的原理及Python中的使用,以及布隆过滤器如何应对Redis中的缓存穿透,相信你对布隆过滤 ... [详细]
  • Spring – Bean Life Cycle
    Spring – Bean Life Cycle ... [详细]
  • 本文介绍了几种常用的图像相似度对比方法,包括直方图方法、图像模板匹配、PSNR峰值信噪比、SSIM结构相似性和感知哈希算法。每种方法都有其优缺点,适用于不同的应用场景。 ... [详细]
  • javascript分页类支持页码格式
    前端时间因为项目需要,要对一个产品下所有的附属图片进行分页显示,没考虑ajax一张张请求,所以干脆一次性全部把图片out,然 ... [详细]
  • 本文详细介绍了 PHP 中对象的生命周期、内存管理和魔术方法的使用,包括对象的自动销毁、析构函数的作用以及各种魔术方法的具体应用场景。 ... [详细]
  • 本文是Java并发编程系列的开篇之作,将详细解析Java 1.5及以上版本中提供的并发工具。文章假设读者已经具备同步和易失性关键字的基本知识,重点介绍信号量机制的内部工作原理及其在实际开发中的应用。 ... [详细]
  • 在尝试对 QQmlPropertyMap 类进行测试驱动开发时,发现其派生类中无法正常调用槽函数或 Q_INVOKABLE 方法。这可能是由于 QQmlPropertyMap 的内部实现机制导致的,需要进一步研究以找到解决方案。 ... [详细]
  • 本文介绍了如何在 Spring 3.0.5 中使用 JdbcTemplate 插入数据并获取 MySQL 表中的自增主键。 ... [详细]
  • IOS Run loop详解
    为什么80%的码农都做不了架构师?转自http:blog.csdn.netztp800201articledetails9240913感谢作者分享Objecti ... [详细]
  • Java高并发与多线程(二):线程的实现方式详解
    本文将深入探讨Java中线程的三种主要实现方式,包括继承Thread类、实现Runnable接口和实现Callable接口,并分析它们之间的异同及其应用场景。 ... [详细]
  • 题目《BZOJ2654: Tree》的时间限制为30秒,内存限制为512MB。该问题通过结合二分查找和Kruskal算法,提供了一种高效的优化解决方案。具体而言,利用二分查找缩小解的范围,再通过Kruskal算法构建最小生成树,从而在复杂度上实现了显著的优化。此方法不仅提高了算法的效率,还确保了在大规模数据集上的稳定性能。 ... [详细]
  • 构建基础的字符串队列实现方法
    在探讨如何构建基础的字符串队列实现方法时,我们发现许多开发者在面对这一问题时常常感到困惑。实际上,队列的基本原理非常简单,即遵循先进先出的原则。然而,在具体实现过程中,需要注意的是Java语言中并没有指针的概念,因此需要通过嵌套类来模拟指针,进而构建链表结构。这种实现方式不仅能够有效地管理字符串数据,还能提升代码的可读性和维护性。 ... [详细]
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社区 版权所有