之前只是用数组对队列进行简单模拟,存在很多问题。这篇博客使用数组模拟循环队列。
- front指针指向队头元素,rear指针指向队尾元素的下一个位置。
- 队列满时,依旧保留一个空闲单元。如下图所示。
- 队列为空的判断条件:rear == front。
- 由于是循环队列,rear可能比front大也可能比其小。所以队列最大尺寸是maxSize时,队列满的判断条件是:(rear + 1) % maxSize == front。取模为了整合rear与front大小为一个问题。
- 当rear>front时,队列长度rear-front。但当rear<front时,队列长度分为两段,一段是maxSize-front,另一段是rear,合起来就是rear-front+maxSize。由此我们可以得到计算队列长度的通用方式为:(rear- front +maxSize ) % maxSize;
代码实现
class CircleQueueDemo{
private int maxSize ;
private int front; //头指针
private int rear; //尾指针
private int[] arr; //存放数组,模拟队列
public CircleQueueDemo(int maxSize) {
this.maxSize = maxSize;
arr = new int[maxSize];
}
public boolean isFull() {
return (rear + 1) % maxSize == front;
}
public boolean isEmpty() {
return rear == front;
}
//添加數據到隊列
public void addQueue(int n) {
if(isFull()) {
System.out.println("队列已满");
return;
}
arr[rear] = n;
//将rear后移动
rear = (rear + 1) % maxSize;
}
public int getQueue(){
if(isEmpty()) {
// System.out.println("队列为空,不能取数据");
throw new RuntimeException("队列为空,不能取数据");
}
int value = arr[front];
front = (front + 1) % maxSize;
return value;
}
public void showQueue() {
if(isEmpty()) {
System.out.println("队列为空");
return;
}
for(int i = front; i < front + size(); i++) {
System.out.println(arr[i % maxSize]);
}
}
public void headQueue() {
if(isEmpty()) {
System.out.println("队列为空,没有数据");
}
System.out.println(arr[front]);
}
public int size() {
return (maxSize - front + rear) % maxSize;
}
}
测试代码如下
import java.util.Scanner;
public class CircleQueue {
public static void main(String[] args) {
CircleQueueDemo queue = new CircleQueueDemo(4);//有效数据最大为3,要保留一个空位置
char key;//接受用户输入
Scanner scanner = new Scanner(System.in);
boolean flag = true;
while(flag) {
System.out.println("s(show)");
System.out.println("e(exit)");
System.out.println("a(add)");
System.out.println("g(get)");
System.out.println("h(head)");
key = scanner.next().charAt(0);
switch(key) {
case 's':
queue.showQueue();
break;
case 'a':
System.out.println("输入一个数");
int value = scanner.nextInt();
queue.addQueue(value);
break;
case 'g':
try {
int res = queue.getQueue();
System.out.println("取出的数据为:" + res);
}catch(Exception e) {
System.out.println(e.getMessage());
}
break;
case 'h':
queue.headQueue();
break;
case 'e':
scanner.close();
flag = false;
break;
}
}
}
}