数据结构与算法--稀疏数组

1.引入

比如在编写五子棋时要实现存盘退出和继续上盘的功能。

如果使用二维数组来记录,每行每列,白子对应2,黑子对应1,默认值对应0.然后这里黑子对应二维数组a[1][2]。白子对应二维数组a[2][3]。

如果棋子很少,那么这么大的棋盘中只有这几个棋子,对应的二维数组中有很多0,可以说是没有意义的数据。那么可以用稀疏数组对这个二维数组进行压缩。

2.基本介绍

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

稀疏数组的处理方法:

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

简单点说就是稀疏数组只记录有数据的行和列。

3.举例:

这里数组中有8个非0数据,有6行7列


[0]这里存放原来数组的行列数以及非0数据的个数。

下面[1]到[8]就对应非0元素的行和列的下标以及值。比如原来的a[0][3]是22,这里就存0,3,22

那么就是9行3列,9*3=27,6*7=42.确实占用空间变少了。

4.再来看开始的棋盘

用稀疏数组的话,可以变成这样

[0]处为11行,11列,2个非元素

第一个非0元素在[1][2]处,row为1,col为2,val为1

第二个非0元素在[2][3]处,row为2,col为3,val为2

原来的二维数组规模为11*11,现在为3*3

5.二维数组与稀疏数组转化思路

要把二维数组转化为稀疏数组:

  • 先遍历原始的二维数组,得到有效的数据的个数sum
  • 再根据这个sum创建稀疏数组sparseArr int[sum+1][3]  (行数不确定,要看有多少个有效数据,再加上第0行的统计这行。列数固定为3
  • 将二维数组的有效数据存入到稀疏数组

要把稀疏数组恢复为二维数组:

  • 先读取稀疏数组的第一行,因为第一行记录了二维数组的行和列。读取后创建原始的二维数组。如chessArr2=int[11][11]
  • 读取后几行的数据,赋值给原始的二维数组。(如果无效值为0的话,因为二维数组没有赋值时也是默认为0,就直接通过sparseArr将二维数组中对应的赋值,就不用多层循环了,像下面这样)

6.代码

public class SparseArray {
    public static void main(String[] args) {
        //创建原始二维数组,0表示默认没有棋子,1表示黑子,2表示白子
        int chessArr1[][]=new int[11][11];
        chessArr1[1][2]=1;
        chessArr1[2][3]=2;
        //用增强for循环输出原始的二维数组
        System.out.println("二维数组");
        for(int[] row : chessArr1){
            for(int data:row){
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }
        //转为稀疏数组,先遍历二维数组得到非0数据个数
        int sum=0;
        for(int i=0;i<11;i++){
            for(int j=0;j<11;j++){
                if(chessArr1[i][j]!=0){
                    sum++;
                }
            }
        }
        System.out.println("非0值有"+sum+"个");
        //创建稀疏数组
        int sparseArr[][]=new int[sum+1][3];
        //给稀疏数组赋值
        sparseArr[0][0]=11;
        sparseArr[0][1]=11;
        sparseArr[0][2]=sum;
        //加个计数器,用来记录是第几个非0元素
        int count=0;
        //遍历二维数组,将非0值存到稀疏数组
        for(int i=0;i<11;i++){
            for(int j=0;j<11;j++){
                if(chessArr1[i][j]!=0){
                    count++;//遇到非0就加一
                    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\t\n",sparseArr[i][0],sparseArr[i][1],sparseArr[i][2]);
        }
        //将稀疏数组恢复成二维数组
        System.out.println("恢复为二维数组");
        int row=sparseArr[0][0];
        int col=sparseArr[0][1];
        int val=sparseArr[0][2];
        //先找到行和列长度
        int chessArr2[][]=new int[row][col];
        //从稀疏数组第二行开始遍历,赋值给二维数组
        for(int t=1;t<=val;t++){
            chessArr2[sparseArr[t][0]][sparseArr[t][1]]=sparseArr[t][2];
        }
        //输出
        for(int[] row1 : chessArr2){
            for(int data:row1){
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }
    }
}

代码中要注意的一点就是我是取t<=val,然后t从1开始。如果是用sparseArr.length的话就是t<sparseArr.length

然后下面这里是增强for

为什么需要增强for循环?

在某些情况下,常规的遍历方式容易显得代码臃肿,增强for可以 简化数组和集合的遍历 , 增强代码的可读性 。

增强for 格式 : for (数据类型 变量名 : 数组或者集合对象) { //循环体 }


运行结果:

成功实现将二维数组转化为稀疏数组,再转化回二维数组


加油加油^_^

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/579581.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

AtCoder Regular Contest 176 C. Max Permutation(计数 分类讨论)

题目 思路来源 乱搞ac 题解 1. 如果有边的权值是1&#xff0c;意味着有两个点的权值都是1&#xff0c;无解 2. 如果一个点i被多个max条件控制&#xff0c;它的值不能超过这些max里最小的那个&#xff0c;记做up[i] 3. 如果同一个权值w对应的边不少于2条&#xff0c;这些边…

Spring Task学习记录

介绍 cron表达式 cron表达式在线生成器 链接: link 入门案例 Component Slf4j public class MyTask {/*** 定时任务 每隔5秒触发1次*/Scheduled(cron "0/5 * * * * ?")public void executeTask(){log.info("定时任务开始执行&#xff1a;{}", new Date…

AtCoder Beginner Contest 173 F - Intervals on Tree(计数 树的性质 贡献)

题目 思路来源 洛谷题解AT_abc173_f Intervals on Tree 题解 - 洛谷专栏 题解 一棵树&#xff0c;考虑加边的过程&#xff0c;加一条边减少一个连通块 那么&#xff0c;逆向这个过程&#xff0c;没删一条边&#xff0c;就多一个连通块 树&#xff1a;点的个数边的个数1 森…

后端端口也可以直接在浏览器访问

比如在浏览器输入http://localhost:8078/hello/helloword访问的是后端的 RestController RequestMapping("/hello") public class HelloWord {RequestMapping("/helloword")public String helloWord(){return "hello word";} }浏览器将会返回

JavaEE——介绍 HTTPServlet 三部分使用与 cookie 和 session 的阐述

文章目录 一、HTTPServlet介绍其中的关键 三个方法 二、HTTPServletRequest(处理请求)1.分块介绍方法作用get 为前缀的方法字段中 含有 getParameter 字段 的方法(前后端交互)&#xff1a;字段中 含有 getHeader 字段 的方法&#xff1a; 2.解释前后端的交互过程3.使用 json 格…

科技感十足特效源码

源码介绍 科技感十足特效源码&#xff0c;源码由HTMLCSSJS组成&#xff0c;记事本打开源码文件可以进行内容文字之类的修改&#xff0c;双击html文件可以本地运行效果&#xff0c;也可以上传到服务器里面 源码截图 源码下载 科技感十足特效源码

Python_AI库 Matplotlib的应用简例:绘制与保存折线图

本文默认读者已具备以下技能&#xff1a; 熟悉Python基础语法&#xff0c;以自行阅读python代码块熟悉Vscode或其它编辑工具的应用 在数据可视化领域&#xff0c;Matplotlib无疑是一个强大的工具。它允许我们创建各种静态、动态、交互式的可视化图形&#xff0c;帮助我们更好…

pyaibote--安卓自动化环境配置与基础的使用方法

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 pyaibote介绍 pyaibote是一个全新&#xff0c;强大的办公自动化库。 支持找图&#xff0c;识别像素等操作。 比appium快十倍。 文章介绍 有大佬给我提到这个库后&#xff0c;我来查看。然后发现这个库太新了&am…

Coursera: An Introduction to American Law 学习笔记 Week 04: Constitutional Law

An Introduction to American Law 本文是 https://www.coursera.org/programs/career-training-for-nevadans-k7yhc/learn/american-law 这门课的学习笔记。 文章目录 An Introduction to American LawInstructors Week 04: Constitutional LawKey Constitutional Law TermsSup…

redission原理笔记

加锁成功的线程&#xff0c;将UUID和线程id和key绑定&#xff0c; 加锁成功后&#xff0c;内部有一个看门狗机制&#xff0c;每隔十秒看下当前线程是否还持有锁&#xff0c;延长生存时间。 没有获取锁的就一直自旋等待&#xff0c;直到超时。 如果redis是主从同步的&#xff0…

Android Studio gradle 默认sourceSets配置

一. AS默认的sourceSets配置 sourceSets在Android插件中如何使用的&#xff1a;android {sourceSets {main {manifest.srcFile AndroidManifest.xmljava.srcDirs [src]resources.srcDirs [src]aidl.srcDirs [src]renderscript.srcDirs [src]res.srcDirs [res]assets.srcD…

Anti Rookit -- 检测隐藏进程

Anti Rookit 一&#xff1a;检测隐藏进程 引言 检测隐藏进程除了众所周知的枚举进程ID之外&#xff0c;还有枚举句柄表的方式。不过今天给大家带来的是第三种方法。 探究 应用层通过接口 C r e a t e P r o c e s s \textcolor{cornflowerblue}{CreateProcess} CreateProcess…

现代信号处理7_最小二乘(CSDN_20240428)

最小二乘法最早由高斯在18世纪提出&#xff0c;几百年以来&#xff0c;这种方法一直被广泛应用。 最小二乘简介 这里是研究最小二乘的起点。随机变量只能存在与理论计算中&#xff0c;我们在工程实践中对随机变量的认识与理论计算中得到的关于随机变量的各种性质相比&#xff…

Penpad 再获 Animoca Brands 投资,全新生态历程

Penpad是Scroll生态的LaunchPad & Yield Aggregator平台&#xff0c;该平台近日在融资上取得了系列进展。据悉&#xff0c;Penpad在前不久率先获得了来自于Gate Labs以及Scroll联合创始人Sandy Peng的融资&#xff0c;并且在近日&#xff0c;其又获得了来自于知名加密投资机…

Coursera: An Introduction to American Law 学习笔记 Week 01: Tort Law

An Introduction to American Law 本文是 https://www.coursera.org/programs/career-training-for-nevadans-k7yhc/learn/american-law 这门课的学习笔记。 文章目录 An Introduction to American LawInstructors SyllabusWeek 01: Tort LawKey Tort Law TermsTort Law: Part …

2024.阳光能源追光计划暨大陆考察团交流分享会

近日大陆考察团抵达香港&#xff0c;受到了本司热情接待和安排。公司于4月27日下午举办了阳光能源追光计划主题交流会。 会上公司营销部总监张超&#xff0c;分享了阳光能源近几年的能源发展之路及公司新推出的追光计划&#xff0c;得到了大陆考察交流团团长杨国均先生的高度赞…

CSS3(响应式布局)

#过渡# 属性连写&#xff1a; transition: width 2s linear 1s; //前一个时间用于表示过渡效果持续时间&#xff0c;后一个时间用于表示过渡效果的延迟。 #转换# #2D转换# 和 #3D转换# 注意&#xff1a;其中angle对应单位为&#xff1a;deg #圆角# #边框# …

java中的泛型(三)——通配符

在前面的文章中我们简要介绍了泛型的概念以及泛型类和泛型方法的使用。在介绍泛型时我们说过在在java中一般用E、T、K、V、N、?这几个字母和符号来表示泛型&#xff0c;对于前面的几个字符它们的使用没有区别&#xff0c;只要注意它们所代表的类型就好。而对于最后一个&#x…

Oracle集群-常用查询及操作(工作日常整理)

1.Oracle集群状态 select * from gv$instance; 示例结果&#xff1a; 2.Oracle集群-增大表空间 常见问题&#xff1a; 导入时或使用时&#xff0c;提示无法extend table ,增加表空间即可 常用操作&#xff1a; 1&#xff09;查询表空间 select * from dba_tablespaces; --…

微信小程序[黑马笔记]

简介 常用组件 视图组件 <!--pages/list/list.wxml--><scroll-view class"container1" scroll-y><view>A</view><view>B</view><view>A</view></scroll-view><!--pages/list2/list.wxml--><swiper …