笔记:用数组创建的队列
昨天忘记发了,就和今天的一起发上来了。
队列
特点
- 有序列表,可以用素组或者链表实现
- 原则是先入先出
使用数组的结构实现队列
- 创建一个类,用来模拟队列,里面需要一个数组,并设置最大容量 maxSize
- 设置两个变量 front 和 rear 分别记录队列的前后端,front 记录队列中先存入的数据的前一个位置(指向队列头部),rear 记录的是后来存入的数据(指向队列的尾部)
- 设置队列类的构造方法,获取 maxSize 的值,创建数组,初始化一些变量
- 添加队列的一些操作方法 
  - 判断队列是否满了 isFull()
- 判断队列是否为空 isEmpty()
- 往队列中添加数据 addQueue()
- 获取队列中的数据 getQueue()
- 查看队列中的所有数据 showQueue()
- 查看队列的头数据 headQueue()
 
例子
package com.atguigu.queue;
import java.util.Scanner;
public class Demo001 {
    public static void main(String[] args) {
        Queue queue = new Queue(3);
        Scanner scanner = new Scanner(System.in);
        char key;
        boolean loop = true;
        while(loop){
            System.out.println("s(show) : 展示列表中的全部元素");
            System.out.println("a(add) : 添加元素到列表中");
            System.out.println("g(get) : 从列表中获取元素");
            System.out.println("h(head) : 查看列表的头部元素");
            System.out.println("e(exit) : 退出系统");
            System.out.print("请输入你的操作代号:");
            key = scanner.next().charAt(0); // 这个地方老师这样写的,真的感觉很妙,能避免误按多一个键发生的错误
            switch(key){
                case 's' :
                    queue.show();
                    break;
                case 'a' :
                    System.out.print("请输入你要添加的数据(int类型):");
                    int value = scanner.nextInt();
                    queue.addQueue(value);
                    break;
                case 'g' :
                    try {
                        queue.getQueue();
                    } catch (Exception e) {
                        System.out.println(e.getMessage());
                    }
                    break;
                case 'h':
                    try {
                        queue.headQueue();
                    } catch (Exception e) {
                        System.out.println(e.getMessage());
                    }
                    break;
                case 'e' :
                    loop = false;
                    break;
                default :
                    break;
            }
        }
        System.out.println("退出系统");
    }
}
// 创建一个类模拟队列
class Queue {
    // 添加一个数组,用来存储数据
    int[] arr;
    // 设置两个变量 rear 和 front 指向队列的末尾和队列头部的前一个位置
    int rear;
    int front;
    // 创建构造方法,在方法内初始化数组和 rear 、 front 的值
    public Queue(int len){
        rear = -1;
        front = -1;
        arr = new int[len];
    }
    // 创建一个判断数组是不是空的方法
    public boolean isEmpty(){
        return rear == front;
    }
    // 创建一个判断数组是不是满的方法
    public boolean isFull(){
        return rear == arr.length - 1;
    }
    // 创建一个展示所有数据的方法
    public void show(){
        if(isEmpty()){
            System.out.println("列表为空,没有数据");
        }
        for (int i = 0; i < rear - front; i++) {
            System.out.println("arr[" + i + "]" + " = " + arr[front + i + 1]);
        }
    }
    // 创建添加数据的方法
    public void addQueue(int num){
        if(isFull()){
            throw new RuntimeException("列表已满,无法添加数据");
        }
        rear++;
        arr[rear] = num;
    }
    // 创建读取数据的方法
    public int getQueue(){
        if(isEmpty()){
            throw new RuntimeException("列表已空,无法获取数据");
        }
        front++;
        return arr[front];
    }
    // 创建获取头部数据的方法
    public void headQueue(){
        if(isEmpty()){
            throw new RuntimeException("列表为空,没有头部数据");
        }
        System.out.println(arr[front + 1]);
    }
}
上面的这个例子有一个缺点,就是整个队列(数组)只能使用一次(front 和 rear 扫过数组长度个元素,),即使队列没满,在使用过一次之后也无法再进行数据的存储等操作了,这样的话其实很浪费内存空间,为了使内存空间的利用率更高,需要将其改造成环形队列,这样就能在队列没满的情况下就往里面添加元素,使内存空间的利用率更高。
改造成环形队列
改造的思路:
- front 指向队列的第一个元素,rear 指向队列的最后一个元素的下一个位置,但它们两个的初始值都为 0
- 队列满了的条件是:( rear + maxSize) % maxSize == front
- 队列为空的条件是: rear == front
- 队列有效数据的个数:( rear + maxSize - front ) % maxSize
例子
package com.atguigu.queue;
import java.util.Scanner;
public class CircleArrayQueueDemo {
    public static void main(String[] args) {
        CircleArray circleArray = new CircleArray(3);
        Scanner scanner = new Scanner(System.in);
        char key;
        boolean loop = true;
        while(loop){
            System.out.println("s(show) : 展示列表中的全部元素");
            System.out.println("a(add) : 添加元素到列表中");
            System.out.println("g(get) : 从列表中获取元素");
            System.out.println("h(head) : 查看列表的头部元素");
            System.out.println("e(exit) : 退出系统");
            System.out.print("请输入你的操作代号:");
            key = scanner.next().charAt(0); // 这个地方老师这样写的,真的感觉很妙,能避免误按多一个键发生的错误
            switch(key){
                case 's' :
                    try {
                        circleArray.show();
                    } catch (Exception e) {
                        System.out.println(e.getMessage());
                    }
                    break;
                case 'a' :
                    System.out.print("请输入你要添加的数据(int类型):");
                    int value = scanner.nextInt();
                    try {
                        circleArray.addQueue(value);
                    } catch (Exception e) {
                        System.out.println(e.getMessage());
                    }
                    break;
                case 'g' :
                    try {
                        circleArray.getQueue();
                    } catch (Exception e) {
                        System.out.println(e.getMessage());
                    }
                    break;
                case 'h':
                    try {
                        System.out.println(circleArray.headQueue());
                    } catch (Exception e) {
                        System.out.println(e.getMessage());
                    }
                    break;
                case 'e' :
                    loop = false;
                    break;
                default :
                    break;
            }
        }
        System.out.println("退出系统");
    }
}
class CircleArray {
    private int maxSize;
    private int rear;
    private int front;
    private int[] arr;
    public CircleArray(int arrMaxSize){
        maxSize = arrMaxSize;
        arr = new int[maxSize];
    }
    public boolean isFull(){
        return (rear + 1) % maxSize == front;
    }
    public boolean isEmpty(){
        return rear == front;
    }
    public void addQueue(int num){
        if(isFull()){
            throw new RuntimeException("列表已满,无法添加数据");
        }
        arr[rear] = num;
        rear = (rear + 1) % maxSize; // 这里妙呀,我还用 if 去判断呢
    }
    public int getQueue(){
        if(isEmpty()){
            throw new RuntimeException("列表为空,无法取出数据");
        }
        int temp = arr[front];
        front = (front + 1) % maxSize;
        return temp;
    }
    public void show(){
        if(isEmpty()){
            throw new RuntimeException("列表为空,没有数据可以展示");
        }
        for(int i = front; i < front + size(); i++){
            System.out.println("arr[" + (i % maxSize) + "] = " + arr[i % maxSize]); 
        }
    }
    public int size(){
        return (rear + maxSize - front) % maxSize;
    }
    public int headQueue(){
        if(isEmpty()){
            throw new RuntimeException("列表为空,没有头数据");
        }
        return arr[front];
    }
}
上面改造的代码最主要的区别就是 rear 和 front 的定义变动,以及为了满足循环队列的要求使用了一些简单的算法。这部分代码出现在方法的返回值以及一些条件判断之中。
今天还学了一点链表,内容的话只有一点,就不和这上面的放在一起了,等链表的内容都学了,放到一起去
参考资料:https://www.bilibili.com/video/BV1E4411H73v?p=14&spm_id_from=333.880.my_history.page.click









