第 3 章:稀疏数组和队列

1、稀疏数组

1.1、实际需求

  • 编写的五子棋程序中,有存盘退出和续上盘的功能
  • 因为该二维数组的很多值是默认值 0 ,因此记录了很多没有意义的数据,我们将其转为稀疏数组进行存储

1.2、稀疏数组应用

1.2.1、稀疏数组处理方法

  • 稀疏数组把具有不同值的元素的行列及值记录在一个小规模的数组中,从而缩小程序的规模
  • 稀疏数组也是二维数组,行数由原数组的数据决定,列数一般为 3 列
  • 稀疏数组的第一行记录原数组一共有几行几列,有多少个不为零的值
    • 第一列:原数组的行数
    • 第二列:原数组的列数
    • 第三列:原数组有多少个不为零的值
  • 之后的行记录原数组中不为零(x)的值所在的行数、列数以及 x 的值
    • 第一列:x 在原数组中的行数
    • 第二列:x 在原数组中的列数
    • 第三列:x 的值

1.2.2、举例说明

  • 原始二维数组较大,压缩后占用空间减少

1.3、应用实例

1.3.1、思路分析

  • 使用稀疏数组, 来保留类似前面的二维数组(棋盘、 地图等等)
  • 把稀疏数组存盘, 并且可以从新恢复原来的二维数组数

1.3.2、代码实现

  • 代码
public class SparseArray {public static void main(String[] args) {// 创建一个原始的二维数组 11 * 11// 0: 表示没有棋子, 1 表示 黑子 2 表蓝子int chessArr1[][] = new int[11][11];chessArr1[1][2] = 1;chessArr1[2][3] = 2;chessArr1[4][5] = 2;// 输出原始的二维数组System.out.println("原始的二维数组~~");for (int[] row : chessArr1) {for (int data : row) {System.out.printf("%d\t", data);}System.out.println();}// 将二维数组 转 稀疏数组的思// 1. 先遍历二维数组 得到非0数据的个数int sum = 0;for (int i = 0; i < chessArr1.length; i++) {for (int j = 0; j < chessArr1[i].length; j++) {if (chessArr1[i][j] != 0) {sum++;}}}// 2. 创建对应的稀疏数组int sparseArr[][] = new int[sum + 1][3];// 给稀疏数组赋值sparseArr[0][0] = chessArr1.length;sparseArr[0][1] = chessArr1[0].length;sparseArr[0][2] = sum;// 遍历二维数组,将非0的值存放到 sparseArr中int count = 0; // count 用于记录是第几个非0数据for (int i = 0; i < chessArr1.length; i++) {for (int j = 0; j < chessArr1[i].length; j++) {if (chessArr1[i][j] != 0) {count++;sparseArr[count][0] = i;sparseArr[count][1] = j;sparseArr[count][2] = chessArr1[i][j];}}}// 输出稀疏数组的形式System.out.println();System.out.println("得到稀疏数组为~~~~");for (int i = 0; i < sparseArr.length; i++) {System.out.printf("%d\t%d\t%d\n", sparseArr[i][0], sparseArr[i][1], sparseArr[i][2]);}System.out.println();// 将稀疏数组 --》 恢复成 原始的二维数组/** 1. 先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组,比如上面的 chessArr2 = int [11][11] 2.* 在读取稀疏数组后几行的数据,并赋给 原始的二维数组 即可.*/// 1. 先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组int chessArr2[][] = new int[sparseArr[0][0]][sparseArr[0][1]];// 2. 在读取稀疏数组后几行的数据(从第二行开始),并赋给 原始的二维数组 即可for (int i = 1; i < sparseArr.length; i++) {chessArr2[sparseArr[i][0]][sparseArr[i][1]] = sparseArr[i][2];}// 输出恢复后的二维数组System.out.println();System.out.println("恢复后的二维数组");for (int[] row : chessArr2) {for (int data : row) {System.out.printf("%d\t", data);}System.out.println();}}}
  • 程序运行结果
原始的二维数组~~
0   0   0   0   0   0   0   0   0   0   0
0   0   1   0   0   0   0   0   0   0   0
0   0   0   2   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   2   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0   得到稀疏数组为~~~~
11  11  3
1   2   1
2   3   2
4   5   2恢复后的二维数组
0   0   0   0   0   0   0   0   0   0   0
0   0   1   0   0   0   0   0   0   0   0
0   0   0   2   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   2   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0   0   0

1.4、课后练习

  • 在前面的基础上, 将稀疏数组保存到磁盘上, 比如 map.data
  • 恢复原来的数组时, 读取 map.data 进行恢复

2、队列

2.1、队列使用场景

  • 银行排队的案例:

2.2、队列介绍

  • 队列是一个有序列表, 可以用数组或是链表来实现。
  • 遵循先入先出的原则, 即: 先存入队列的数据, 要先取出,后存入的要后取出
  • 示意图: (使用数组模拟队列示意图)

2.3、数组模拟队列

2.3.1、思路分析

  • maxSize :队列容量(数组的长度)
  • arr :模拟队列的数组
  • front :指向队列头部元素的前一个元素,初始值为 -1
  • rear :指向队列尾部元素,初始值为 -1

  • 基本操作

    • 队列判空:front == rear
    • 队列判满:rear == (maxSize - 1) ,即 rear 是否已经指向了数组的最后一个位置
    • 队列元素个数:rear - front
    • 队列入队:队列不满才能入队,arr[++rear] = value
    • 队列出队:队列不空才能出队,return arr[front++]

2.3.2、代码实现

  • 队列的定义
// 使用数组模拟队列-编写一个ArrayQueue类
class ArrayQueue {private int maxSize; // 表示数组的最大容量private int front; // 队列头private int rear; // 队列尾private int[] arr; // 该数据用于存放数据, 模拟队列// 创建队列的构造器public ArrayQueue(int arrMaxSize) {maxSize = arrMaxSize;arr = new int[maxSize];front = -1; // 指向队列头部,分析出front是指向队列头的前一个位置.rear = -1; // 指向队列尾,指向队列尾的数据(即就是队列最后一个数据)}// 判断队列是否满public boolean isFull() {return rear == maxSize - 1;}// 判断队列是否为空public boolean isEmpty() {return rear == front;}// 添加数据到队列public void addQueue(int n) {// 判断队列是否满if (isFull()) {System.out.println("队列满,不能加入数据~");return;}rear++; // 让 rear 后移arr[rear] = n;}// 获取队列的数据, 出队列public int getQueue() {// 判断队列是否空if (isEmpty()) {// 通过抛出异常throw new RuntimeException("队列空,不能取数据");}front++; // front后移return arr[front];}// 显示队列的所有数据public void showQueue() {// 遍历if (isEmpty()) {System.out.println("队列空的,没有数据~~");return;}for (int i = front + 1; i <= rear; i++) {// Java 中也能用占位符诶System.out.printf("arr[%d]=%d\n", i, arr[i]);}}// 显示队列的头数据, 注意不是取出数据public int headQueue() {// 判断if (isEmpty()) {throw new RuntimeException("队列空的,没有数据~~");}return arr[front + 1];}
}
  • 测试代码
public class ArrayQueueDemo {public static void main(String[] args) {// 测试一把// 创建一个队列ArrayQueue queue = new ArrayQueue(3);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): 查看队列头的数据");System.out.println();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.printf("取出的数据是%d\n", res);} catch (Exception e) {System.out.println(e.getMessage());}break;case 'h': // 查看队列头的数据try {int res = queue.headQueue();System.out.printf("队列头的数据是%d\n", res);} catch (Exception e) {System.out.println(e.getMessage());}break;case 'e': // 退出scanner.close();loop = false;break;default:break;}}System.out.println("程序退出~~");}}
  • 程序运行结果
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据s
队列空的,没有数据~~
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
1
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
2
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据s
arr[0]=1
arr[1]=2
arr[2]=3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
4
队列满,不能加入数据~
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
取出的数据是1
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
取出的数据是2
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
取出的数据是3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
队列空,不能取数据
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据

2.4、数组模型环形队列

2.4.1、提出问题

  • 目前数组使用一次就不能用, 没有达到复用的效果,造成内存空间的浪费
  • 将这个数组使用算法, 改进成一个环形的队列(取模: %

2.4.2、思路分析

  • 对前面的队列进行优化,改造为环形队列(通过取模实现)
  • maxSize :队列容量(数组的长度)
  • arr :模拟队列的数组
  • front :指向队列头部元素,初始值为 0
  • rear :指向队列尾部元素的后一个元素,初始值为 0

  • 基本操作

    • 队列判空:front == rear
    • 队列判满:
      • 为何要在 rear 之后,front 之前空出一个元素的空间?因为如果不空出一个元素,队列判空条件为:front == rear ,队列判满的条件也是:front == rear ,有歧义
      • 队列容量:因为空出了一个元素,所以队列容量就变成了 (maxSize - 1)
      • 当空出一个元素的空间,如何判满?当还剩一个元素时,队列就已经满了,所以判断条件为 (rear + 1) % maxSize == front
    • 队列元数个数:
      • 计算公式:(rear + maxSize - front) % maxSize ,这样来思考:
      • 当 rear 比 front 大时,即 (rear -front) > 0 ,这时还没有形成环形结构,(rear -front) 即是队列元素个数
      • 当 rear 比 front 小时,即 (rear -front) < 0 ,这时已经形成了环形结构,(rear -front) 表示数组还差多少个元素存满(负数),(rear + maxSize - front) 即是队列元素个数
      • 综上:(rear + maxSize - front) % maxSize
    • 队列入队:
      • 首先,队列不满才能入队
      • 由于 rear 指向队列尾部元素的后一个元素,所以直接设置即可: arr[rear] = value
      • 接下来,rear 应该向后移动一个位置:rear = (rear + 1) % maxSize
      • 取模是为了防止数组越界,让指针从新回到数组第一个元素
    • 队列出队:
      • 首先,队列不空才能出队
      • 由于 front 直接指向队列头部元素,所以直接返回该元素即可:int value = arr[front ]
      • 接下来,front 应该向后移动一个位置:front = (front + 1) % maxSize
      • 取模是为了防止数组越界,让指针从新回到数组第一个元素

2.4.3、代码实现

  • 环形队列的实现
class CircleArray {private int maxSize; // 表示数组的最大容量// front 变量的含义做一个调整: front 就指向队列的第一个元素, 也就是说 arr[front] 就是队列的第一个元素// front 的初始值 = 0private int front;// rear 变量的含义做一个调整:rear 指向队列的最后一个元素的后一个位置. 因为希望空出一个空间做为约定.// rear 的初始值 = 0private 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 = 3return (rear + maxSize - front) % maxSize;}// 显示队列的头数据, 注意不是取出数据public int headQueue() {// 判断if (isEmpty()) {throw new RuntimeException("队列空的,没有数据~~");}return arr[front];}
}
  • 测试代码
public class CircleArrayQueueDemo {public static void main(String[] args) {// 测试一把System.out.println("测试数组模拟环形队列的案例~~~");// 创建一个环形队列CircleArray queue = new CircleArray(4); // 说明设置4, 其队列的有效数据最大是3char 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): 查看队列头的数据");System.out.println();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.printf("取出的数据是%d\n", res);} catch (Exception e) {// TODO: handle exceptionSystem.out.println(e.getMessage());}break;case 'h': // 查看队列头的数据try {int res = queue.headQueue();System.out.printf("队列头的数据是%d\n", res);} catch (Exception e) {// TODO: handle exceptionSystem.out.println(e.getMessage());}break;case 'e': // 退出scanner.close();loop = false;break;default:break;}}System.out.println("程序退出~~");}}
  • 程序运行结果
测试数组模拟环形队列的案例~~~
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
1
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
2
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据s
arr[0]=1
arr[1]=2
arr[2]=3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据a
输出一个数
4
队列满,不能加入数据~
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
取出的数据是1
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
取出的数据是2
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据s
arr[2]=3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
取出的数据是3
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据g
队列空,不能取数据
s(show): 显示队列
e(exit): 退出程序
a(add): 添加数据到队列
g(get): 从队列取出数据
h(head): 查看队列头的数据

第 3 章:稀疏数组和队列相关推荐

  1. 【Java数据结构与算法】第一章 稀疏数组和队列

    第一章 稀疏数组和队列 文章目录 第一章 稀疏数组和队列 一.线性结构和非线性结构 1.线性结构 2.非线性结构 二.稀疏数组 三.队列 1.队列 2.环形队列 一.线性结构和非线性结构 1.线性结构 ...

  2. 002稀疏数组和队列[超大章]

    稀疏数组和队列 稀疏数组 假设我们有一个棋盘 我们可以看到,现在只有两个棋子在我们的棋盘上,虽然我们可以完全使用二维数组来保存整张棋盘,但是这样实在是太浪费空间了,这个时候我们就需要稀疏数组 当一个数 ...

  3. 【自学笔记】尚硅谷数据结构与算法Chapter 2 稀疏数组和队列

    Chapter 2 稀疏数组和队列 文章目录 Chapter 2 稀疏数组和队列 2.1 稀疏数组 2.1.1 基本介绍 2.1.2 应用案例 2.2 队列 2.2.1 队列介绍 2.2.2 用数组模 ...

  4. 新星计划Day4【数据结构与算法】 稀疏数组与队列

    新星计划Day4[数据结构与算法] 稀疏数组与队列

  5. 数组结构与算法-007-015稀疏数组与队列

    007 稀疏数组的应用场景 基本介绍: 当一个数组中大部分元素为0,或者为同一个数值的数组时,可以稀疏数组来保存该数组 稀疏数组的处理方法是: 记录数组一共有几行几列,有多少个不同的值 把具有不同值的 ...

  6. Qz学算法-数据结构篇(稀疏数组、队列)

    目录 1.稀疏(sparse array)数组 需求引入 分析问题 2.队列(queue) 1.数组模拟队列 需求引入 1.1介绍 1.2数组模拟队列 1.3思路分析 1.4代码实现 2.数组模拟环形 ...

  7. 数据结构之你没见过的稀疏数组和队列刨析!

    大家好!,我是小刘,很长一段时间,没更新了,今天和大家复习一下,数据结构中的稀疏数组,仅作分享,一起交流,哈哈! 1.稀疏数组 1.1.实际需求 编写的五子棋程序中,有存盘退出和续上盘的功能 因为该二 ...

  8. golang数据结构与算法——稀疏数组、队列和链表

    文章目录 一 稀疏数组 1.1 先看一个实际的需求 1.2 稀疏数组基本介绍 1.3 稀疏数组举例说明 1.4 把数组转换为稀疏数组实现 1.5 把稀疏数组还原为原数组 二 队列 2.1 队列的介绍 ...

  9. 单片机实现环形队列_稀疏数组和队列(二)

    队列的介绍 队列以一种先入先出(FIFO)的线性表,还有一种先入后出的线性表(FILO)叫做栈. 教科书上有明确的定义与描述.类似于现实中排队时的队列(队尾进,队头出),队列只在线性表两端进行操作,插 ...

最新文章

  1. 初三女生学计算机专业,江西中专初三毕业学计算机专业适宜女生吗
  2. Object Hook 简单介绍
  3. TypeScript 素描 - 函数
  4. 数组的升序 java_java – 以升序数组排序数组
  5. (51)SSDT HOOK 实现进程保护
  6. 戏说 Windows GDI (2)
  7. 洛阳理工Linux实验报告,洛阳理工学院实验报告.doc
  8. vue省市区三级联动mysql,js/json,html/jsp
  9. linux删除文件操作
  10. 用注册表禁止windows添加新用户
  11. 计算机科学与技术指导情况记录表,计算机科学与技术系 教学情况检查表(期初)...
  12. axios_的其他方式发送请求_使用axios.request .get .delete .post .put 等方法发送请求---axios工作笔记005
  13. 留言板分页php,php留言板代码[经典的分页代码](1/4)
  14. OpenShift免费空间绑定顶级域名(图文教程)
  15. java并发编程(4)--单例模式的安全问题 volatile
  16. socket网络字节序以及大端序小端序
  17. socket通信原理_Appium原理
  18. linux安装CUPS详细教程,ubuntu安装打印机CUPS
  19. [转载]使用 Apache Geronimo 和 JMS 构建事件驱动的框架
  20. 查看已下载的Docker镜像latest具体版本

热门文章

  1. 专业零售商行业调研报告 - 市场现状分析与发展前景预测(2021-2027年)
  2. ssm如何支持热部署_IntelliJ IDEA基于SpringBoot如何搭建SSM开发环境
  3. android开发学术报告,基于Android应用的安全防护技术研究
  4. Wi-Fi 6还没用上,Wi-Fi 7就要来了?
  5. Linux 吉祥物30岁了!“出道”30年,你对Linux了解多少?
  6. 一次解决Linux内核内存泄漏实战全过程
  7. SolarWinds 升级 APM Suite,简化应用程序和基础架构管理!
  8. Rust 让人奔溃的那些特性!
  9. 半小时训练亿级规模知识图谱,亚马逊这个 AI 框架要火!
  10. 达摩院成立XG实验室!阿里官宣进军5G