之前已经做过队列的学习笔记,这一篇写的是循环队列,大部分代码可以继续沿用,某些地方需要作出更改,使其可以实现循环队列的功能。
通俗的总结一下队列的操作,我的思路是将头指针固定不动,然后每一次元素入队就将尾指针后移,每一次出队就把当前头指针指向的元素返回,然后将整个数组整体前移一个位置,尾指针同时减一。而循环队列可以看作将数组围成一个圆形,头指针和尾指针都会根据入队与出队而发生改变,这样就可以省去队列数据结构出队时需要将出队元素除外的其他所有元素整体前移,从而增加了效率。
创建一个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重叠时,即证明当前队列为空。(队列已满应该为两指针相邻,所以才有判满的那条公式。这也是循环数组需要预留一个位置用于判断,否则判满与判空的条件很可能就相同从而无法判断了。)
public boolean isEmpty(){
return rear == front;
}
四、元素入队
入队前,判满操作还是不能少。通过判断后,将出入元素element保存在rear所指向的位置,然后rear指针按照公式移动(为了实现循环,需要取模操作)。最后打印“入队成功提示”作提示。
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;
}
//peek
public void rearQueue(){
if(isEmpty()){
System.out.println("队列为空,peek不了啦~");
return;
}
System.out.println("队头元素是:" + arr[front]);
}
//
}