一、问题分析并优化:
- 目前数组队列使用一次就不能用了,没有达到复用的效果
- 将这个数组使用算法:改进成一个环形的队列,取模 %
二、思路:
- 将front变量的含义做调整:front变为指向队列的第一个元素,也就是说array[front]是队列的第一个元素,front的初始值为0
- 将rear变量的含义做调整:rear变为指向队列的最后一个元素的后一个位置,因为希望空出一个空间作为约定。rear的初始值为0
- 当队列满使得条件是:(rear +1)%maxSize == front
- 当队列为空的条件是:rear==front
- 当我们这样分析的时候,队列中有效的个数:(rear+maxSize-front)% maxSize
三、代码实现
class CircleArray{
private int maxSize;
private int front;
private int rear;
private int[] array;
//提供一个构造器
public CircleArray(int arrMaxSize){
maxSize = arrMaxSize;
array = new int[maxSize];
}
//判断队列是否满了
public boolean isFull(){
return (rear +1) % maxSize == front;
}
//判断队列是否为空
public Boolean isNull(){
return rear == front;
}
//往数组环形队列添加数据
public void addElement(int n){
if(isFull()){
System.out.println("队列已满,禁止添加数据!");
return;
}
array[rear] = n; //直接将数据加入
rear = (rear+1) % maxSize; //
}
//出队列
public int getElement(){
if(isNull()){
throw new RuntimeException("队列为空,没有数据");
}
//这里需要分析出front是指向队列的第一个元素
//1.先把front对应的值保存在一个临时变量中,
//2.将front后移,考虑取模
//3.将临时保存的变量返回
int value = array[front];
front = (front+1)%maxSize;
return value;
}
//显示所有数据
public void showQueue(){
if(isNull()){
System.out.println("队列中没有数据!");
return;
}
//从front开始遍历,遍历多少个元素:数组队列中有效个数为:(rear+maxSize-front)%maxSize
int v = (rear+maxSize-front)%maxSize;
for(int i = front;i < front+v;i++){
System.out.println(array[i]);
}
}
//显示头部数据
public int showHeadQueue(){
if(isNull()){
throw new RuntimeException("队列中没有数据");
}
return array[front];
}
}
测试类:
//用数组模拟队列--->环形队列
public class ArrayQueueDemo2 {
public static void main(String[] args){
//测试
//创建一个数组模拟的环形队列
CircleArray queue = new CircleArray(4);
char key = ' ';
Scanner scanner = new Scanner(System.in);
boolean loop = true;
//输出一个菜单
while(loop){
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.addElement(value);
break;
case 'g':
try {
int res = queue.getElement();
System.out.println("取出的数据为"+res);
}catch (Exception exception){
System.out.println(exception.getMessage());
}
break;
case 'h':
try{
int head = queue.showHeadQueue();
System.out.println("头部的数据为"+head);
}catch(Exception e){
System.out.println(e.getMessage());
}
break;
case 'e':
scanner.close();
loop = false;
break;
default:
break;
}
}
System.out.println("程序退出!");
}
}
测试得知:可以实现程序复用,