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

Java队列Queue及循环队列

队列是一种线性结构。相比数组,队列对应的操作是数组的子集。只能从一段(队尾)添加元素,只能从另一端(队首取出元素)。 队列的操作:          

队列是一种线性结构。

相比数组,队列对应的操作是数组的子集。

只能从一段(队尾)添加元素,只能从另一端(队首取出元素)

 

队列的操作:

                        《Java 队列Queue及循环队列》

 

队列的实现:

《Java 队列Queue及循环队列》

添加元素(入队)      : void enqueue(E)
删除、取出元素(出队)      :  E dequeue()。
取得队首元素             :E getFront()
查看队列有多少元素  :int getSize()
判断队列是否为空      :boolean isEmpty()

Queue接口:

public interface Queue {
int getSize();
boolean isEmpty();
void enqueue(E e);
E dequeue();
E getFront();
}

这里我们将复用之前我的一篇文章中讲到的数组类Array,这里就不进行展示了。有兴趣可以了解动态数组的操作:

  • https://blog.csdn.net/qq_41723615/article/details/88413667
  • https://blog.csdn.net/qq_41723615/article/details/85456789

 

ArrayQueue:基于Array类实现的队列

public class ArrayQueue implements Queue {
//创建私有Array对象
private Array array;
//基于动态数组的实现,传入容量变量
public ArrayQueue(int capacity){
//知道要容量为多少,在这里由用户设置
array = new Array<>(capacity);
}
public ArrayQueue(){
//不知道容量为多少
array = new Array<>();
}
@Override
public int getSize(){
return array.getSize();
}
@Override
public boolean isEmpty(){
return array.isEmpty();
}
public int getCapacity(){
return array.getCapacity();
}
@Override
public void enqueue(E e){
//基于动态数组实现的队列,如果容量不足,则会触发动态数组方法进行扩容
array.addLast(e);
}
@Override
public E dequeue(){
//相应的此操作也会触发减容的操作
return array.removeFirst();
}
@Override
public E getFront(){
return array.getFirst();
}
@Override
public String toString(){
StringBuilder res = new StringBuilder();
res.append("Queue: ");
res.append("front [");
//遍历array元素,逐次加入队列中
for(int i = 0 ; i res.append(array.get(i));
if(i != array.getSize() - 1) {
res.append(", ");
}
}
res.append("] tail");
return res.toString();
}
public static void main(String[] args) {
ArrayQueue queue = new ArrayQueue<>();
for(int i = 0 ; i <10 ; i ++) {
queue.enqueue(i);
System.out.println(queue);
if(i % 3 == 2) {
queue.dequeue();
System.out.println(queue);
}
}
}
}

打印结果:

《Java 队列Queue及循环队列》

下面对数组队列的操作,进行复杂度分析:

《Java 队列Queue及循环队列》

取出队列元素时间复杂度为O(n),每取出一个元素,后面的元素都要做相应的位置调整。怎么优化它呢?

没错,可以利用循环队列的实现。

《Java 队列Queue及循环队列》

《Java 队列Queue及循环队列》

当队列为空时,front和tail指向同一个地方。

当插入一个元素时:只需要维护tail这个位置

《Java 队列Queue及循环队列》

《Java 队列Queue及循环队列》

当出队一个元素时:只需要改变front的位置即可

《Java 队列Queue及循环队列》

《Java 队列Queue及循环队列》

当tail加满了怎么办?

《Java 队列Queue及循环队列》

此时可以把此数组队列看成一个环行结构。7位置紧接着0这个位置,两个位置是相连的,也就是环形结构。

《Java 队列Queue及循环队列》

那么当tail插入满了的时候,就可以重新利用前面取出的空间进行下面的操作了。

tail的计算公式:tail =(当前的索引+1)% (整个数组的长度)

当再入队一个元素时:

《Java 队列Queue及循环队列》

tail则继续 +1即可。

《Java 队列Queue及循环队列》

此时有一个空出来的位置,那么此位置下还能入队一个元素吗?

答案是不能,队列满的设计是:   《Java 队列Queue及循环队列》

前面已经给出了队列空的设计:《Java 队列Queue及循环队列》

如果再加入一个元素,则会与我们前面的设计产生矛盾。其实整个设计操作,就和时钟一样。

 

循环队列的实现:LoopQueue

此时不再复用之前实现的动态数组,这里将重新进行逻辑设计:

public class LoopQueue implements Queue {
//定义一个数组
private E[] data;
//定义用于描述队首,队尾的变量
private int front, tail;
//定义成员变量,描述队列有多少个元素
private int size;
//用户希望数组队列要传入多少元素
public LoopQueue(int capacity){
//capacity + 1:队列需要空出一个位置
data = (E[])new Object[capacity + 1];
//空队列初始化成员变量
frOnt= 0;
tail = 0;
size = 0;
}
//用户不知道要设置队列容量为多少时,新建队列默认容量为10
public LoopQueue(){
this(10);
}
//获取队列有多少个元素的个数
public int getCapacity(){
return data.length - 1;
}
@Override
public boolean isEmpty(){
return frOnt== tail;
}
@Override
public int getSize(){
return size;
}
//入队
@Override
public void enqueue(E e){
//根据设计要求,判断队列是否满了
if((tail + 1) % data.length == front) {
//扩容
resize(getCapacity() * 2);
}
//存放元素
data[tail] = e;
//当存放元素后,需要设置循环队列tail
tail = (tail + 1) % data.length;
size ++;
}
//出队
@Override
public E dequeue(){
if(isEmpty()) {
throw new IllegalArgumentException("Cannot dequeue from an empty queue.");
}
// 出队元素为当前队首元素
E ret = data[front];
data[front] = null;
frOnt= (front + 1) % data.length;
size --;
if(size == getCapacity() / 4 && getCapacity() / 2 != 0) {
//缩容
resize(getCapacity() / 2);
}
return ret;
}
//取得队首元素
@Override
public E getFront(){
if(isEmpty()) {
throw new IllegalArgumentException("Queue is empty.");
}
return data[front];
}
//扩容
private void resize(int newCapacity){
//创建新数组队列
E[] newData = (E[])new Object[newCapacity + 1];
for(int i = 0 ; i //把旧队列元素放入新队列空间
//偏移位置量:front
newData[i] = data[(i + front) % data.length];
}
data = newData;
frOnt= 0;
//扩容不影响原队列的队首、队尾的位置
tail = size;
}
//打印队列
@Override
public String toString(){

StringBuilder res = new StringBuilder();
res.append(String.format("Queue: size = %d , capacity = %d\n", size, getCapacity()));
res.append("front [");
//由于是循环操作,所以tail值可能比front小
for(int i = front ; i != tail ; i = (i + 1) % data.length){
res.append(data[i]);
//判断是否为队列最后一个元素
if((i + 1) % data.length != tail) {
res.append(", ");
}
}
res.append("] tail");
return res.toString();
}
public static void main(String[] args){
LoopQueue queue = new LoopQueue<>();
for(int i = 0 ; i <10 ; i ++) {
queue.enqueue(i);
System.out.println(queue);
if(i % 3 == 2) {
queue.dequeue();
System.out.println(queue);
}
}
}
}

打印结果:

Queue: size = 1 , capacity = 10
front [0] tail
Queue: size = 2 , capacity = 10
front [0, 1] tail
Queue: size = 3 , capacity = 10
front [0, 1, 2] tail
Queue: size = 2 , capacity = 5
front [1, 2] tail
Queue: size = 3 , capacity = 5
front [1, 2, 3] tail
Queue: size = 4 , capacity = 5
front [1, 2, 3, 4] tail
Queue: size = 5 , capacity = 5
front [1, 2, 3, 4, 5] tail
Queue: size = 4 , capacity = 5
front [2, 3, 4, 5] tail
Queue: size = 5 , capacity = 5
front [2, 3, 4, 5, 6] tail
Queue: size = 6 , capacity = 10
front [2, 3, 4, 5, 6, 7] tail
Queue: size = 7 , capacity = 10
front [2, 3, 4, 5, 6, 7, 8] tail
Queue: size = 6 , capacity = 10
front [3, 4, 5, 6, 7, 8] tail
Queue: size = 7 , capacity = 10
front [3, 4, 5, 6, 7, 8, 9] tail

下面对循环队列做时间复杂度分析:

《Java 队列Queue及循环队列》

虽然循环队列的实现比较复杂,不过相比性能考虑,是值得这么做的。

 

下面对数组队列及循环队列的性能进行对比:

import java.util.Random;
public class Main {
// 测试使用q运行opCount个enqueueu和dequeue操作所需要的时间,单位:秒
private static double testQueue(Queue q, int opCount){
long startTime = System.nanoTime();
Random random = new Random();
//入队
for(int i = 0 ; i q.enqueue(random.nextInt(Integer.MAX_VALUE));
}
//出队
for(int i = 0 ; i q.dequeue();
}
long endTime = System.nanoTime();
return (endTime - startTime) / 1000000000.0;
}
public static void main(String[] args) {
int opCount = 100000;
//数组队列
ArrayQueue arrayQueue = new ArrayQueue<>();
double time1 = testQueue(arrayQueue, opCount);
System.out.println("ArrayQueue, time: " + time1 + " s");
//循环队列
LoopQueue loopQueue = new LoopQueue<>();
double time2 = testQueue(loopQueue, opCount);
System.out.println("LoopQueue, time: " + time2 + " s");
}
}

 

结果:《Java 队列Queue及循环队列》

忽略硬件与JVM版本的影响。知道了队列的具体实现,之后,我需要知道队列其实对于队首可以多种定义,根据队列的应用场景的不同而定义。队列还分广义的队列,我实现的只是普通的队列,不过对于这种普通队列的实现,在应用算法的时候,有它重要的用处,例如广度优先遍历的应用,后续将会对广度优先遍历做进一步探究。


推荐阅读
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 微软头条实习生分享深度学习自学指南
    本文介绍了一位微软头条实习生自学深度学习的经验分享,包括学习资源推荐、重要基础知识的学习要点等。作者强调了学好Python和数学基础的重要性,并提供了一些建议。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 本文介绍了闭包的定义和运转机制,重点解释了闭包如何能够接触外部函数的作用域中的变量。通过词法作用域的查找规则,闭包可以访问外部函数的作用域。同时还提到了闭包的作用和影响。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 原文地址:https:www.cnblogs.combaoyipSpringBoot_YML.html1.在springboot中,有两种配置文件,一种 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 推荐系统遇上深度学习(十七)详解推荐系统中的常用评测指标
    原创:石晓文小小挖掘机2018-06-18笔者是一个痴迷于挖掘数据中的价值的学习人,希望在平日的工作学习中,挖掘数据的价值, ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 本文详细解析了JavaScript中相称性推断的知识点,包括严厉相称和宽松相称的区别,以及范例转换的规则。针对不同类型的范例值,如差别范例值、统一类的原始范例值和统一类的复合范例值,都给出了具体的比较方法。对于宽松相称的情况,也解释了原始范例值和对象之间的比较规则。通过本文的学习,读者可以更好地理解JavaScript中相称性推断的概念和应用。 ... [详细]
author-avatar
mobiledu2502910233
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有