数据结构:循环队列
admin
2024-01-22 21:15:01
0

之前已经做过队列的学习笔记,这一篇写的是循环队列,大部分代码可以继续沿用,某些地方需要作出更改,使其可以实现循环队列的功能。

通俗的总结一下队列的操作,我的思路是将头指针固定不动,然后每一次元素入队就将尾指针后移,每一次出队就把当前头指针指向的元素返回,然后将整个数组整体前移一个位置,尾指针同时减一。而循环队列可以看作将数组围成一个圆形,头指针和尾指针都会根据入队与出队而发生改变,这样就可以省去队列数据结构出队时需要将出队元素除外的其他所有元素整体前移,从而增加了效率。


创建一个CircleQueue类,rear表示尾指针,front表示头指针,size表示用于实现循环队列的数组大小,而arr就是用于实现循环队列的数组。

class CircleQueue{private int rear;private int front;private int size;private int[] arr;
}

一、构造循环队列

与普通队列不同的是,头指针front与尾指针rear的初始值都设置为0,并且由于循环队列中需要空出一个位置用于判断和保持操作一致,因此数组实际大小应该比用户期待队列的大小大1,因此size的值应该为用户期待队列大小的值maxsize加一。

    public CircleQueue(int maxsize){size = maxsize + 1;arr = new int[size];rear = 0;front = 0;}

二、判断循环队列是否已满

由于队列是循环的,因此判满的方法也需要更改。当尾指针+1的值对队列大小size取模的结果值刚好为头指针的值则代表队列已满。(用文字描述难免有些晦涩,在草稿画个环作数组运算一下便于理解。)

    public boolean isFull(){return (rear + 1) % size == front;}

三、判断循环队列是否已空

当尾指针rear与头指针front重叠时,即证明当前队列为空。(队列已满应该为两指针相邻,所以才有判满的那条公式(rear+1)%size=front。这也是循环数组需要预留一个位置用于判断,否则判满与判空的条件很可能就相同从而无法判断了。)

    public boolean isEmpty(){return rear == front;}

四、元素入队

入队前,判满操作还是不能少。通过判断后,将出入元素element保存在rear所指向的位置,然后rear指针按照公式(rear +1) % size移动(为了实现循环,需要取模操作)。最后打印“入队成功提示”作提示。

    public void addQueue(int element){if(isFull()){//throw new RuntimeException("队列已满!");System.out.println("队列已满!");return;}arr[rear] = element;rear = (rear +1) % size;System.out.println("入队成功!");}

五、元素出队

出队前同样需要判空操作。接着先将出队元素用临时变量ans保存着,然会按公式移动front指针,让front后退一个位置,最后返回出队元素。

    public int getQueue(){if(isEmpty()){throw new RuntimeException("队列已空!");}int ans = arr[front];front = (front + 1) % size;return ans;}

六、展示队列

循环队列的展示队列操作有点点绕。第一步判空操作(可有可无)。getSzie方法就是计算从rear到size之间的距离,因为是循环队列,所以也需要进行取模运算。计算后,在showQueue方法中,以front为起点,直至front加上相距距离。但是在打印时不能用i作为数组下标,这样很可能会出现数组越界。为了使下标正确,还需要对i进行取模运算,运算结果才是正确的数组下标。最后实现遍历循环队列,输出打印。

    public void showQueue(){if(isEmpty()){System.out.println("队列为空,没有数据~");return;}System.out.println("队列展示:");for(int i = front ; i < front + getSize(); i++){System.out.println(arr[i % size]);//System.out.println(arr[i]);}
//        System.out.println("front:" + front);
//        System.out.println("getsize:" + getSize());}public int getSize(){return (rear + size -front) % size;}

七、获取队首元素

这个方法与普通队列中完全相似,同样先判空,若符合要求则返回队首元素。在此不再赘述。

    public void rearQueue(){if(isEmpty()){System.out.println("队列为空,peek不了啦~");return;}System.out.println("队头元素是:" + arr[front]);}

八、界面

界面保持一致,不需要作更改。将代码再放一次在这里。

public class CircleArray {public static void main(String[] args){Scanner input = new Scanner(System.in);boolean flag = true;char key = ' ';System.out.print("请输入队列大小:");Scanner inputs = new Scanner(System.in);int size = inputs.nextInt();CircleQueue AQ = new CircleQueue(size);while(flag){System.out.println("---------------------------");System.out.println("|(1)退出程序:e(exit)     |");System.out.println("|(2)元素入队:a(add)      |");System.out.println("|(3)元素出队:g(get)      |");System.out.println("|(4)查询队头:p(peek)     |");System.out.println("|(5)展示队列:s(show)     |");System.out.println("---------------------------");System.out.print("【输入字母选择操作】");key = input.next().charAt(0);switch (key){case 'a':System.out.print("入队元素:");int element = input.nextInt();AQ.addQueue(element);break;case 'g':System.out.println("出队元素是:" + AQ.getQueue());break;case 'p':AQ.rearQueue();break;case 's':AQ.showQueue();break;case 'e':System.out.println("成功退出...");flag = false;break;default:System.out.println("输入不合法,请重新输入~");}}}}

完整代码:

import java.util.Scanner;public class CircleArray {public static void main(String[] args){Scanner input = new Scanner(System.in);boolean flag = true;char key = ' ';System.out.print("请输入队列大小:");Scanner inputs = new Scanner(System.in);int size = inputs.nextInt();CircleQueue AQ = new CircleQueue(size);while(flag){System.out.println("---------------------------");System.out.println("|(1)退出程序:e(exit)     |");System.out.println("|(2)元素入队:a(add)      |");System.out.println("|(3)元素出队:g(get)      |");System.out.println("|(4)查询队头:p(peek)     |");System.out.println("|(5)展示队列:s(show)     |");System.out.println("---------------------------");System.out.print("【输入字母选择操作】");key = input.next().charAt(0);switch (key){case 'a':System.out.print("入队元素:");int element = input.nextInt();AQ.addQueue(element);break;case 'g':System.out.println("出队元素是:" + AQ.getQueue());break;case 'p':AQ.rearQueue();break;case 's':AQ.showQueue();break;case 'e':System.out.println("成功退出...");flag = false;break;default:System.out.println("输入不合法,请重新输入~");}}}}class CircleQueue{private int rear;private int front;private int size;private int[] arr;//构造public CircleQueue(int maxsize){size = maxsize + 1;arr = new int[size];rear = 0;front = 0;}//判满public boolean isFull(){return (rear + 1) % size == front;}//判空public boolean isEmpty(){return rear == front;}//入队public void addQueue(int element){if(isFull()){//throw new RuntimeException("队列已满!");System.out.println("队列已满!");return;}arr[rear] = element;rear = (rear +1) % size;System.out.println("入队成功!");}//出队public int getQueue(){if(isEmpty()){throw new RuntimeException("队列已空!");}int ans = arr[front];front = (front + 1) % size;return ans;}//展示队列public void showQueue(){if(isEmpty()){System.out.println("队列为空,没有数据~");return;}System.out.println("队列展示:");for(int i = front ; i < front + getSize(); i++){System.out.println(arr[i % size]);//System.out.println(arr[i]);}
//        System.out.println("front:" + front);
//        System.out.println("getsize:" + getSize());}public int getSize(){return (rear + size -front) % size;}//peekpublic void rearQueue(){if(isEmpty()){System.out.println("队列为空,peek不了啦~");return;}System.out.println("队头元素是:" + arr[front]);}//}

相关内容

热门资讯

自己制作安卓系统教程,自制安卓... 亲爱的读者们,你是否曾梦想过摆脱安卓系统的束缚,亲手打造一个只属于你自己的操作系统?别再羡慕那些技术...
安卓系统调整器下载,轻松优化手... 你有没有发现,手机用久了,系统总是有点小问题,比如卡顿啦,电池续航不给力啦,这些小烦恼是不是让你头疼...
怎样升级安卓系统视频,安卓系统... 亲爱的手机控们,你是否也和我一样,对手机系统升级充满了好奇和期待?想象你的安卓手机在经过一番“变身”...
鸿蒙系统和安卓系统哪个广告少,... 你有没有发现,现在手机市场上的操作系统真是五花八门,让人挑花了眼。不过,最近有个话题特别火,那就是鸿...
安卓系统openrec怎么注册... 你有没有想过,想要在安卓系统上体验一把OpenRec的乐趣,却发现注册步骤有点让人摸不着头脑?别急,...
怎样更新车机安卓系统,车机安卓... 亲爱的车主朋友们,你是不是也和我一样,对车机系统里的安卓系统充满了好奇?想要让它焕然一新,变得更加强...
安卓机换系统后照片,照片如何完... 你有没有遇到过这种情况:手机里的照片突然消失得无影无踪,心里那个急啊,就像热锅上的蚂蚁。别担心,今天...
安卓手机定位系统软件,技术原理... 你有没有想过,你的安卓手机里那些神奇的定位系统软件是怎么工作的呢?它们就像你的私人侦探,随时随地告诉...
安卓制作win系统盘,打造Wi... 亲爱的读者,你是否曾想过,将安卓系统的魅力与Windows系统的强大功能完美结合?今天,就让我带你一...
系统警告_您的安卓手机,揭秘潜... 亲爱的手机主人,最近你的安卓手机是不是突然跳出来一个系统警告,让你心头一紧?别慌,今天就来给你详细解...
投屏安卓系统版本,揭秘不同版本... 你有没有想过,家里的电视屏幕那么大,却只能用它来看电视?现在,有了安卓系统,你就可以把手机上的精彩内...
安卓官方系统升级软件,畅享智能... 你有没有发现,你的安卓手机最近是不是变得有点儿“年轻”了?没错,这就是安卓官方系统升级的魅力所在!今...
安卓系统铃声长度是多少,时长差... 你有没有想过,为什么你的手机每次收到消息时,都会响起那熟悉的铃声?是不是好奇过,安卓系统的铃声长度到...
酷派电脑系统安卓,深度解析与全... 亲爱的读者们,你是否曾对那些在电脑世界里游刃有余的酷派电脑系统安卓版心生好奇?今天,就让我带你一起揭...
什么系统可以装安卓软件,基于A... 你有没有想过,你的手机里那些好玩又实用的安卓软件,其实也可以在其他设备上运行呢?没错,这就是今天我们...
制作安卓系统主题软件,安卓系统... 你有没有想过,给你的安卓手机换一个全新的面貌?没错,就是那种一打开手机,就能感受到完全不同的风格和氛...
安卓系统平板怎么截屏,操作指南... 亲爱的平板用户,你是不是也和我一样,有时候想记录下平板上的精彩瞬间,却发现截屏功能有点神秘?别担心,...
安卓系统不推送更新,揭秘背后的... 最近是不是发现你的安卓手机有点儿“懒”啊?更新推送总是慢吞吞的,让人等得花儿都谢了。别急,今天就来给...
ape格式转换安卓系统,享受音... 你有没有想过,你的安卓手机里的ape格式音乐文件,竟然可以通过一个小小的转换,焕发出全新的生命力?没...
获取安卓系统加载器,核心功能与... 你有没有想过,你的安卓手机里那些神奇的软件和游戏是怎么被安装到你的设备上的呢?没错,就是通过一个叫做...