在编程中,算法的重要性不言而喻,没有算法的程序是没有灵魂的。可见算法的重要性。

然而,在学习算法之前我们需要掌握数据结构,数据结构是算法的基础。

我在大学的时候,学校里的数据结构是用C语言教的,因为对C语言也不是很了解,所以掌握得不是特别好,在网上找的一些学习资料里也基本都是用C语言来进行数据结构的教学。

那么,从本篇文章开始,我将用Java语言来介绍数据结构,当然,数据结构过后就是算法。

线性结构和非线性结构
  1. 线性结构

    线性结构作为最常用的数据结构,其特点是数据元素之间存在一对一的线性关系;

    线性结构有两种不同的存储结构,即顺序存储结构和链式存储结构。顺序存储的线性表称为顺序表,顺序表中存储的元素是连续的;

    链式存储的线性表称为链表,链表中存储的元素不一定是连续的,元素节点中存放数据元素以及相邻元素的地址信息;

    线性结构常见的有:数组、队列、链表和栈
  2. 非线性结构

    非线性结构包括:二维数组、多维数组、广义表、树结构、图结构
稀疏数组

对数据结构有了一个初步的认识之后,我们开始对一些具体的数据结构进行详细的分析。

我们来看一个实际的需求:

这是一个五子棋的程序,有存盘退出和续上盘的功能,如下图,如何将下图的棋局进行保存呢?



那这个问题很简单,很多人可能会想到用二维数组来进行存储。



如上图,我们用0表示无子,1表示黑子,2表示蓝子,但是这个程序问题很大,因为该二维数组的很多值都是默认值0,因此记录了很多没有意义的数据,那么这个时候我们就可以使用稀疏数组来对该二维数组进行一个压缩。

那么稀疏数组到底是什么呢?

当一个数组中大部分元素是0,或者为同一个值的数组时,可以使用稀疏数组来保存该数组。

稀疏数组的处理方法是:

  1. 记录数组一共有几行几列,有多少个不同的值
  2. 把具有不同值的元素的行列以及值记录在一个小规模的数组中,从而缩小程序的规模

那么了解了稀疏数组的概念后,我们通过稀疏数组来改进一下五子棋程序。



经过稀疏数组的压缩之后,原数组从原来的11行11列变为了三行三列。

该稀疏数组的第一行记录的是原数组的行数和列数以及元素个数。

接下来的每一行记录的是有效元素的位置和值,例如第二行记录的是原数组中位于1,2位置上的元素1;第三行记录的是原数组中位于2,3位置上的元素2。

综上所述,二维数组转稀疏数组的思路:

  1. 遍历原始的二维数组,得到要保存的有效元素个数
  2. 根据有效元素个数创建稀疏数组sparseArr
  3. 将二维数组的有效数据存入稀疏数组即可

稀疏数组转原始二维数组的思路:

  1. 先读取稀疏数组的第一行,根据第一行的数据创建原始二维数组
  2. 读取稀疏数组后几行的数据,并赋给原始的二维数组即可

关于实现思路已经分析完毕,接下来用代码实现。

将二维数组转稀疏数组用代码实现如下:

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; System.out.println("原始的二维数组:"); for (int[] row : chessArr1) {
for (Integer value : row) {
System.out.printf("%d\t", value);
}
System.out.println();
} // 将二维数组转稀疏数组
// 先遍历二维数组,得到非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++;
}
}
} // 创建对应的稀疏数组
int sparseArr[][] = new int[sum + 1][3];
// 给稀疏数组赋值
// 稀疏数组第一行存的是原始数组的行数、列数和有效元素个数
sparseArr[0][0] = chessArr1.length;
sparseArr[0][1] = chessArr1[0].length;
sparseArr[0][2] = sum; // 遍历二维数组,将非0的值存入到稀疏数组中
int count = 0; // 用于记录是第几个非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[] row : sparseArr) {
for (Integer value : row) {
System.out.printf("%d\t", value);
}
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 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 0 稀疏数组:
11 11 2
1 2 1
2 3 2

这样,我们就成功地将二维数组转为了稀疏数组。

那么用代码如何将稀疏数组转为二维数组呢?

		// 将稀疏数组转为二维数组
// 先读取稀疏数组的第一行,根据第一行的数据创建原始数组
int chessArr2[][] = new int[sparseArr[0][0]][sparseArr[0][1]]; // 读取稀疏数组后几行数据(从第二行开始读取),并赋给原始数组
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 (Integer value : row) {
System.out.printf("%d\t", value);
}
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 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 0 稀疏数组:
11 11 2
1 2 1
2 3 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 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 0

整体代码如下:

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; System.out.println("原始的二维数组:"); for (int[] row : chessArr1) {
for (Integer value : row) {
System.out.printf("%d\t", value);
}
System.out.println();
} // 将二维数组转稀疏数组
// 先遍历二维数组,得到非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++;
}
}
} // 创建对应的稀疏数组
int sparseArr[][] = new int[sum + 1][3];
// 给稀疏数组赋值
// 稀疏数组第一行存的是原始数组的行数、列数和有效元素个数
sparseArr[0][0] = chessArr1.length;
sparseArr[0][1] = chessArr1[0].length;
sparseArr[0][2] = sum; // 遍历二维数组,将非0的值存入到稀疏数组中
int count = 0; // 用于记录是第几个非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[] row : sparseArr) {
for (Integer value : row) {
System.out.printf("%d\t", value);
}
System.out.println();
} // 将稀疏数组转为二维数组
// 先读取稀疏数组的第一行,根据第一行的数据创建原始数组
int chessArr2[][] = new int[sparseArr[0][0]][sparseArr[0][1]]; // 读取稀疏数组后几行数据(从第二行开始读取),并赋给原始数组
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 (Integer value : row) {
System.out.printf("%d\t", value);
}
System.out.println();
}
}

图解Java数据结构之稀疏数组的更多相关文章

  1. Java数据结构之稀疏数组(Sparse Array)

    1.需求 编写的五子棋程序中,有存盘退出和续上盘的功能.因为该二维数组的很多值是默认值0,因此记录了很多没有意义的数据,为了压缩存储所以采用稀疏数组. 2.基本介绍 当一个数组中大部分元素为0,或者为 ...

  2. 图解 Java 数据结构

    图解Java数据结构: 一.链表       Java ListNode     https://www.cnblogs.com/easyidea/p/13371863.html 二.栈       ...

  3. 数据结构(1):稀疏数组使用java实现

    主要是用于数组压缩,去除无效的数组内容: 原数组内容: 0 0 0 0 0 0 1 0 0 0 0 0 0 2 0 0 0 0 0 0 0 0 0 0 0 转换成 稀疏数组 5 5 2 1 1 1 2 ...

  4. Java数据结构和算法 - 数组

    Q: 数组的创建? A: Java中有两种数据类型,基本类型和对象类型,在许多编程语言中(甚至面向对象语言C++),数组也是基本类型.但在Java中把数组当做对象来看.因此在创建数组时,必须使用new ...

  5. golang数据结构之稀疏数组

    掌握知识: 数组的初始化和赋值 结构体的初始化和赋值 字符串和整型之间的转换以及其它的一些操作 类型断言 读取文件 写入文件 对稀疏数组进行压缩 package main import ( " ...

  6. JAVA数据结构--ArrayList动态数组

    在计算机科学中,动态数组,可扩展数组,可调整数组,动态表,可变数组或数组列表是一种随机存取可变大小列表数据结构,允许添加或删除元素.它提供许多现代主流编程语言的标准库.动态数组克服了静态数组的限制,静 ...

  7. (二)Java数据结构和算法——数组

    一.数组的实现 上一篇博客我们介绍了一个数据结构必须具有以下基本功能: ①.如何插入一条新的数据项 ②.如何寻找某一特定的数据项 ③.如何删除某一特定的数据项 ④.如何迭代的访问各个数据项,以便进行显 ...

  8. 图解Java数据结构之队列

    本篇文章,将对队列进行一个深入的解析. 使用场景 队列在日常生活中十分常见,例如:银行排队办理业务.食堂排队打饭等等,这些都是队列的应用.那么队列有什么特点呢? 我们知道排队的原则就是先来后到,排在前 ...

  9. java数据结构系列之——数组(1)

    import javax.management.RuntimeErrorException; public class MyArray { private long array[]; private ...

随机推荐

  1. .net core 发布IIS 出现Http 500错误

    首先再webconfig中设置stdoutLogEnabled="true",然后运行之后,到logs中查看登陆错误日志. 根据不同的错误进行解决: 我的错误是发布文件夹中缺少Dw ...

  2. SSDB数据库笔记

    目录 环境 配置文件 启动服务器 客户端 SSDB:一个高性能的支持丰富数据结构的 NoSQL 数据库, 用于替代 Redis. 参考文献: SSDB官网 环境 win10 下 wsl 环境 ubun ...

  3. CodeForces - 1251C (思维+贪心+归并排序)

    题意 https://vjudge.net/problem/CodeForces-1251C 一个字符串,相邻的偶数奇数不能交换位置,其他相邻的情况可以交换,问字符串代表的数最小是多少. 思路 相邻的 ...

  4. firefox56 版本中的 Selenium IDE 无法导出脚本问题

    firefox:56 Slenium IDE :3系列 问题:Selenium IDE 没有工具栏,无法导出录制的脚本,这给自动化测试工作带来了极大的不便. 解决办法:将firefox 降级 (只有5 ...

  5. glibc-static

    yum install glibc-static yum install libstdc++-static

  6. ipvsadm用法

    其实LVS的本身跟iptables很相似,而且连命令的使用格式都很相似,其实LVS是根据iptables的框架开发的,那么LVS的本身分成了两个部分: 第一部分是工作在内核空间的一个IPVS的模块,其 ...

  7. Pwn-pwn-100

    题目地址http://www.whalectf.xin/files/2779dd8a2562a1d5653c5c6af9791711/binary_100 32位 ,没有防护 上IDA 很简单的栈溢出 ...

  8. day46_9_5前端(3)

    一.调节长宽. 在css中可以对块级标签设置长和宽,但是对行内标签无效,其属性如下: 1.height:80px 高度. 2.width:80px 宽度. 二.字体属性. 设置一个标签中的字体.比如黑 ...

  9. shiro异步请求返回JSON响应

    shiro异步请求返回JSON响应 需求1:当shiro请求资源,但是没有进行认证时,默认是进行重定向,现在需要返回JSON响应.注意异步请求,服务器重定向后,ajax拿到的是浏览器重定向后的到的页面 ...

  10. es6 的类 class

    1.ES6提供了更接近传统语言的写法,引入了Class(类)这个概念,作为对象的模板.通过class关键字,可以定义类. 2. //定义类 class Point { constructor(x, y ...