稀疏数组
当一个数组中大部分元素为0,或者为同一个值的数组时,可以使用稀疏数组来保存该数组。
比如下面这个 10 x 10 的数组中只有三个数据,大部分元素是0。
现在把这个数组改成稀疏数组,如下
可见稀疏数组将原始 10 x 10 的数组变成了 4 x 3 的数组,节省了大量空间。
下面表格解释一下这个稀疏数组(此处行和列中的值都是原始数组的下标,是从0开始的):
稀疏数组(sparse array) | 行(row) | 列(col) | 值(value) |
---|---|---|---|
[0](原始数组规模) | 10(原始数组有10行) | 10(原始数组有10列) | 3(原始数组有3个数据) |
[1](第一个数据) | 1 | 2 | 1 |
[2](第二个数据) | 2 | 5 | 2 |
[3](第三个数据) | 7 | 4 | 5 |
稀疏数组的处理方法是:
- 记录数组一共有几行几列,有几个数据。
- 把有效元素的行列及值记录在一个小规模的数组中,从而缩小程序的规模。
接下来看实现代码:
public class SpareArray {
public static void main(String[] args) {
// 构造原始数组
int row = 10, col = 10;
int [][] arr = new int[row][col];
arr[1][2] = 1;
arr[2][5] = 2;
arr[7][4] = 5;
// 打印原始数组
System.out.println("------------------原始数组---------------");
for (int[] i : arr) {
for (int j : i)
System.out.printf("%d\t", j);
System.out.println();
}
// 遍历原始数组,计算非 0 个数
int count = 0;
for (int i = 0; i < row; i++) {
for (int j = 0; j < col; j++)
if (arr[i][j] != 0)
count++;
}
// 构造稀疏数组
int [][] sparseArr = new int[count+1][3];
// 原始数组转化稀疏数组
sparseArr[0][0] = row;
sparseArr[0][1] = col;
sparseArr[0][2] = count;
int index = 0;
for (int i = 0; i < row; i++) {
for (int j = 0; j < col; j++)
if (arr[i][j] != 0) {
index++;
sparseArr[index][0] = i;
sparseArr[index][1] = j;
sparseArr[index][2] = arr[i][j];
}
}
// 打印稀疏数组
System.out.println("------------------稀疏数组---------------");
for (int[] i : sparseArr) {
for (int j : i)
System.out.printf("%d\t", j);
System.out.println();
}
}
}
运行结果
------------------原始数组---------------
0 0 0 0 0 0 0 0 0 0
0 0 1 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 5 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
------------------稀疏数组---------------
10 10 3
1 2 1
2 5 2
7 4 5
明白了原始数组转化成稀疏数组,那么将稀疏数组恢复为原始数组就很简单了,代码如下:
// 将稀疏数组 sparseArr 恢复成原始数组(和上面道理一样咯)
int [][] arr2 = new int[sparseArr[0][0]][sparseArr[0][1]];
for (int i = 1; i < sparseArr.length; i++)
arr2[sparseArr[i][0]][sparseArr[i][1]] = sparseArr[i][2];
// 打印恢复的数组
System.out.println("------------------恢复数组---------------");
for (int[] i : arr2) {
for (int j : i)
System.out.printf("%d\t", j);
System.out.println();
}
运行结果
------------------恢复数组---------------
0 0 0 0 0 0 0 0 0 0
0 0 1 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 5 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