数据结构 队列

队列介绍

  1. 队列是一个有序列表,可以用数组或是链表来实现。
  2. 遵循先入先出的原则。即:先存入队列的数据,要先取出。后存入的要后取出

image.png

数组模拟队列

数组模拟队列思路

  1. 队列本身是有序列表,若使用数组的结构来存储队列的数据,则队列数组的声明如下图,其中maxSize 是该队列的最大容量。
  2. 因为队列的输出、输入是分别从前后端来处理,因此需要两个变量 front 及 rear 分别记录队列前后端的下标,front 会随着数据输出而改变,而 rear 则是随着数据输入而改变,如图所示:

image.png

  1. 将数据存入队列时称为”addQueue”,addQueue 的处理需要有两个步骤:思路分析

1) 将尾指针往后移:rear+1,当 front == rear 【空】
2) 若尾指针 rear 小于队列的最大下标 maxSize-1,则将数据存入 rear 所指的数组元素中,否则无法存入数据。rear == maxSize - 1[队列满]

代码实现

  1. import java.util.Scanner;
  2. public class ArrayQueueDemo {
  3. public static void main(String[] args) {
  4. //创建一个队列
  5. ArrayQueue queue = new ArrayQueue(3);
  6. char key = ' '; //接收用户输入
  7. Scanner scanner = new Scanner(System.in);//
  8. boolean loop = true;
  9. //输出一个菜单
  10. while(loop) {
  11. System.out.println("s(show): 显示队列");
  12. System.out.println("e(exit): 退出程序");
  13. System.out.println("a(add): 添加数据到队列");
  14. System.out.println("g(get): 从队列取出数据");
  15. System.out.println("h(head): 查看队列头的数据");
  16. key = scanner.next().charAt(0);//接收一个字符
  17. switch (key) {
  18. case 's':
  19. queue.showQueue();
  20. break;
  21. case 'a':
  22. System.out.println("输出一个数");
  23. int value = scanner.nextInt();
  24. queue.addQueue(value);
  25. break;
  26. case 'g': //取出数据
  27. try {
  28. int res = queue.getQueue();
  29. System.out.printf("取出的数据是%d\n", res);
  30. } catch (Exception e) {
  31. // TODO: handle exception
  32. System.out.println(e.getMessage());
  33. }
  34. break;
  35. case 'h': //查看队列头的数据
  36. try {
  37. int res = queue.headQueue();
  38. System.out.printf("队列头的数据是%d\n", res);
  39. } catch (Exception e) {
  40. // TODO: handle exception
  41. System.out.println(e.getMessage());
  42. }
  43. break;
  44. case 'e': //退出
  45. scanner.close();
  46. loop = false;
  47. break;
  48. default:
  49. break;
  50. }
  51. }
  52. System.out.println("程序退出~~");
  53. }
  54. }
  55. // 使用数组模拟队列-编写一个ArrayQueue类
  56. class ArrayQueue {
  57. private int maxSize; // 表示数组的最大容量
  58. private int front; // 队列头
  59. private int rear; // 队列尾
  60. private int[] arr; // 该数据用于存放数据, 模拟队列
  61. // 创建队列的构造器
  62. public ArrayQueue(int arrMaxSize) {
  63. maxSize = arrMaxSize;
  64. arr = new int[maxSize];
  65. front = -1; // 指向队列头部,分析出front是指向队列头的前一个位置.
  66. rear = -1; // 指向队列尾,指向队列尾的数据(即就是队列最后一个数据)
  67. }
  68. // 判断队列是否满
  69. public boolean isFull() {
  70. return rear == maxSize - 1;
  71. }
  72. // 判断队列是否为空
  73. public boolean isEmpty() {
  74. return rear == front;
  75. }
  76. // 添加数据到队列
  77. public void addQueue(int n) {
  78. // 判断队列是否满
  79. if (isFull()) {
  80. System.out.println("队列满,不能加入数据~");
  81. return;
  82. }
  83. rear++; // 让rear 后移
  84. arr[rear] = n;
  85. }
  86. // 获取队列的数据, 出队列
  87. public int getQueue() {
  88. // 判断队列是否空
  89. if (isEmpty()) {
  90. // 通过抛出异常
  91. throw new RuntimeException("队列空,不能取数据");
  92. }
  93. front++; // front后移
  94. return arr[front];
  95. }
  96. // 显示队列的所有数据
  97. public void showQueue() {
  98. // 遍历
  99. if (isEmpty()) {
  100. System.out.println("队列空的,没有数据~~");
  101. return;
  102. }
  103. for (int i = 0; i < arr.length; i++) {
  104. System.out.printf("arr[%d]=%d\n", i, arr[i]);
  105. }
  106. }
  107. // 显示队列的头数据, 注意不是取出数据
  108. public int headQueue() {
  109. // 判断
  110. if (isEmpty()) {
  111. throw new RuntimeException("队列空的,没有数据~~");
  112. }
  113. return arr[front + 1];
  114. }
  115. }

数组模拟环形队列实现

对数组模拟队列的优化,充分利用数组。因此将数组看做是一个环形的。(通过取模的方式来实现即可)

数组模拟环形队列思路

  1. 尾索引的下一个为头索引时表示队列满,即将队列容量空出一个作为约定,这个在做判断队列满的时候需要注意 (rear + 1) % maxSize == front 满]
  2. rear == front [空]

算法思路:

  1. front变量的含义做一个调整: front 就指向队列的第一个元素,也就是说 arr[front] 就是队列的第一个元素 front 的初始值 = 0
  2. rear 变量的含义做一个调整:rear指向队列的最后一个元素的后一个位置。因为希望空出一个空间做为约定。rear 的初始值 = 0
  3. 当队列满时,条件是 (rear + 1) % maxSize == front 【满】
  4. 队列为空的条件,rear == front
  5. 当这样分析,队列中有效的数据的个数 (rear + maxSize - front) % maxSize // rear = 1 front = 0
  6. 在原来的队列上修改得到,一个环形队列

    代码实现

    ```java public class CircleArrayQueueDemo {

    public static void main(String[] args) {

    1. System.out.println("测试数组模拟环形队列的案例~~~");
    2. // 创建一个环形队列
    3. CircleArray queue = new CircleArray(4); //说明设置4, 其队列的有效数据最大是3
    4. char key = ' '; // 接收用户输入
    5. Scanner scanner = new Scanner(System.in);//
    6. boolean loop = true;
    7. // 输出一个菜单
    8. while (loop) {
    9. System.out.println("s(show): 显示队列");
    10. System.out.println("e(exit): 退出程序");
    11. System.out.println("a(add): 添加数据到队列");
    12. System.out.println("g(get): 从队列取出数据");
    13. System.out.println("h(head): 查看队列头的数据");
    14. key = scanner.next().charAt(0);// 接收一个字符
    15. switch (key) {
    16. case 's':
    17. queue.showQueue();
    18. break;
    19. case 'a':
    20. System.out.println("输出一个数");
    21. int value = scanner.nextInt();
    22. queue.addQueue(value);
    23. break;
    24. case 'g': // 取出数据
    25. try {
    26. int res = queue.getQueue();
    27. System.out.printf("取出的数据是%d\n", res);
    28. } catch (Exception e) {
    29. // TODO: handle exception
    30. System.out.println(e.getMessage());
    31. }
    32. break;
    33. case 'h': // 查看队列头的数据
    34. try {
    35. int res = queue.headQueue();
    36. System.out.printf("队列头的数据是%d\n", res);
    37. } catch (Exception e) {
    38. // TODO: handle exception
    39. System.out.println(e.getMessage());
    40. }
    41. break;
    42. case 'e': // 退出
    43. scanner.close();
    44. loop = false;
    45. break;
    46. default:
    47. break;
    48. }
    49. }
    50. System.out.println("程序退出~~");

    }

}

class CircleArray { private int maxSize; // 表示数组的最大容量 //front 变量的含义做一个调整: front 就指向队列的第一个元素, 也就是说 arr[front] 就是队列的第一个元素 //front 的初始值 = 0 private int front; //rear 变量的含义做一个调整:rear 指向队列的最后一个元素的后一个位置. 因为希望空出一个空间做为约定. //rear 的初始值 = 0 private int rear; // 队列尾 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 n) {
    // 判断队列是否满
    if (isFull()) {
        System.out.println("队列满,不能加入数据~");
        return;
    }
    //直接将数据加入
    arr[rear] = n;
    //将 rear 后移, 这里必须考虑取模
    rear = (rear + 1) % maxSize;
}

// 获取队列的数据, 出队列
public int getQueue() {
    // 判断队列是否空
    if (isEmpty()) {
        // 通过抛出异常
        throw new RuntimeException("队列空,不能取数据");
    }
    // 这里需要分析出 front是指向队列的第一个元素
    // 1. 先把 front 对应的值保留到一个临时变量
    // 2. 将 front 后移, 考虑取模
    // 3. 将临时保存的变量返回
    int value = arr[front];
    front = (front + 1) % maxSize;
    return value;

}

// 显示队列的所有数据
public void showQueue() {
    // 遍历
    if (isEmpty()) {
        System.out.println("队列空的,没有数据~~");
        return;
    }
    // 思路:从front开始遍历,遍历多少个元素
    // 动脑筋
    for (int i = front; i < front + size() ; i++) {
        System.out.printf("arr[%d]=%d\n", i % maxSize, arr[i % maxSize]);
    }
}

// 求出当前队列有效数据的个数
public int size() {
    // rear = 2
    // front = 1
    // maxSize = 3 
    return (rear + maxSize - front) % maxSize;   
}

// 显示队列的头数据, 注意不是取出数据
public int headQueue() {
    // 判断
    if (isEmpty()) {
        throw new RuntimeException("队列空的,没有数据~~");
    }
    return arr[front];
}

} ```