【数据结构与算法】稀疏数组

news2024/12/22 19:19:20

文章目录

  • 一:为什么会使用稀疏数组
    • 1.1 先看一个实际的需求
    • 1.2 基本介绍
      • 1.2.1 稀疏数组的处理方法
      • 1.2.2 数组的举例说明
      • 1.2.3 应用实例
      • 1.2.4 整体思路分析
        • 二维数组转稀疏数组的思路
        • 稀疏数组转原始的二维数组的思路
  • 二:代码实现
    • 2.1 创建一个原始的11*11二维数组
    • 2.2 将二维数组转化为稀疏数组
    • 2.3 将稀疏数组恢复成原始的二维数组
    • 2.4 完整代码奉上

一:为什么会使用稀疏数组

1.1 先看一个实际的需求

  • 编写的五子棋程序中,有存盘退出和续上盘的功能
    在这里插入图片描述
  • 问题分析:因为该二维数组的很多值是默认值0, 因此记录了很多没有意义的数据。所以此时我们可以简化一下二维数组,使用稀疏数组来解决此问题。

1.2 基本介绍

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

1.2.1 稀疏数组的处理方法

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

1.2.2 数组的举例说明

在这里插入图片描述
注:

  1. 数组的下标都是从0开始,所以第一行,第四列用(0,3)表示
  2. 第一行【0】6 7 8 表示,共有6行、7列,有效数据有8个,其他行表示这8个数据具体的坐标。

1.2.3 应用实例

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

1.2.4 整体思路分析

在这里插入图片描述

二维数组转稀疏数组的思路

  1. 遍历原始的二维数组,得到有效数据的个数sum
  2. 根据sum就可以创建稀疏数组(sparseArr ) int[sum+1][3]
  3. 将二维数组的有效数据数据存入到 稀疏数组

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

  1. 先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组,比如上面的 chessArr2=int[11][11]
  2. 在读取稀疏数组后几行的数据,并赋给 原始的二维数组 即可

二:代码实现

2.1 创建一个原始的11*11二维数组

0:表示没有棋子 1:表示黑子 2:表示白子

		int[][] chessArr1 = new int[11][11];
        chessArr1[1][2] = 1;
        chessArr1[2][3] = 2;
        chessArr1[4][9] = 3;
        chessArr1[6][3] = 4;
        chessArr1[9][10] = 5;
        //输出二维数组
        System.out.println("原始的二维数组");
        for (int[] row : chessArr1) {
            for (int data : row) {
                //换行输出
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }

在这里插入图片描述

2.2 将二维数组转化为稀疏数组

  1. 遍历原始的二维数组,得到有效数据的个数sum
int sum = 0;
        for (int i = 0; i < chessArr1.length; i++) {
            for (int j = 0; j < chessArr1[0].length; j++) {
                if(chessArr1[i][j] != 0) {
                    sum = sum + 1;
                }
            }
        }
        System.out.println("sum: " + sum);

输出:sum=5

  1. 创建对应的稀疏数组
int[][] sparseArray = new int[sum + 1][3];
  1. 给稀疏数组第一行赋值
		sparseArray[0][0] = chessArr1.length;
        sparseArray[0][1] = chessArr1[0].length;
        sparseArray[0][2] = sum;
  1. 给第其他行赋值,遍历二维数组将非0值存放到稀疏数组里面
//count用于记录第几个非0数据
        int count = 1;
        for (int i = 0; i < chessArr1.length; i++) {
            for (int j = 0; j < chessArr1[0].length; j++) {
                if(chessArr1[i][j] != 0) {
                    sparseArray[count][0] = i;
                    sparseArray[count][1] = j;
                    sparseArray[count][2] = chessArr1[i][j];
                    count = count + 1;
                }
            }
        }

  1. 输出稀疏数组
		System.out.println();
        System.out.println("得到的稀疏数组");
        for (int[] row : sparseArray) {
            for (int data : row) {
                //换行输出
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }

在这里插入图片描述

2.3 将稀疏数组恢复成原始的二维数组

  1. 先读取稀疏数组的第一行。根据第一行的数据,创建二维数组
int[][] chessArr2 = new int[sparseArray[0][0]][sparseArray[0][1]];
  1. 读取稀疏数组的后几行数据(从第二行开始),并赋值给原始的二维数组即可
for (int i = 1; i < sparseArray.length; i++) {
            chessArr2[sparseArray[i][0]][sparseArray[i][1]] = sparseArray[i][2];
        }
        System.out.println("转化后的的二维数组");
        for (int[] row : chessArr1) {
            for (int data : row) {
                //换行输出
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }

在这里插入图片描述

2.4 完整代码奉上

package com.sysg.dataStructuresAndAlgorithms.array;

import java.util.Arrays;

/**
 * 稀疏数组
 *
 * 一:二维数组转稀疏数组的思路
 * 1.遍历原始的二维数组,得到有效数据的个数sum
 * 2.根据sum就可以创建稀疏数组(sparseArr ) int[sum+1][3]
 *
 *
 * 二:将二维数组的有效数据数据存入到 稀疏数组
 * 1.稀疏数组转原始的二维数组的思路
 * 2.先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组,比如上面的 chessArr12=int[11][11]
 * 3.在读取稀疏数组后几行的数据,并赋给 原始的二维数组 即可
 *
 * @author shi.guangqiang
 */
public class SparseArray {

    public static void main(String[] args) {
        //1.创建一个原始的二维数组 11*11
        //0:表示没有棋子 1:表示黑子 2:表示白子
        int[][] chessArr1 = new int[11][11];
        chessArr1[1][2] = 1;
        chessArr1[2][3] = 2;
        chessArr1[4][9] = 3;
        chessArr1[6][3] = 4;
        chessArr1[9][10] = 5;
        //输出二维数组
        System.out.println("原始的二维数组");
        for (int[] row : chessArr1) {
            for (int data : row) {
                //换行输出
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }

         //2.将二维数组转化为稀疏数组的思路
         //2.1 遍历原始的二维数组,得到有效数据的个数sum
        int sum = 0;
        for (int i = 0; i < chessArr1.length; i++) {
            for (int j = 0; j < chessArr1[0].length; j++) {
                if(chessArr1[i][j] != 0) {
                    sum = sum + 1;
                }
            }
        }
        System.out.println("sum: " + sum);

        //2.2 创建对应的稀疏数组
        int[][] sparseArray = new int[sum + 1][3];

        //2.3 给稀疏数组赋值
        //2.3.1 给第一行赋值
        sparseArray[0][0] = chessArr1.length;
        sparseArray[0][1] = chessArr1[0].length;
        sparseArray[0][2] = sum;
        //2.3.2 给第其他行赋值,遍历二维数组将非0值存放到稀疏数组里面
        //count用于记录第几个非0数据
        int count = 1;
        for (int i = 0; i < chessArr1.length; i++) {
            for (int j = 0; j < chessArr1[0].length; j++) {
                if(chessArr1[i][j] != 0) {
                    sparseArray[count][0] = i;
                    sparseArray[count][1] = j;
                    sparseArray[count][2] = chessArr1[i][j];
                    count = count + 1;
                }
            }
        }

        //3.输出稀疏数组
        System.out.println();
        System.out.println("得到的稀疏数组");
        for (int[] row : sparseArray) {
            for (int data : row) {
                //换行输出
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }

        //4.将稀疏数组恢复成原始的二维数组
        //4.1 先读取稀疏数组的第一行。根据第一行的数据,创建二维数组
        int[][] chessArr2 = new int[sparseArray[0][0]][sparseArray[0][1]];
        //4.2 读取稀疏数组的后几行数据(从第二行开始),并赋值给原始的二维数组即可
        for (int i = 1; i < sparseArray.length; i++) {
            chessArr2[sparseArray[i][0]][sparseArray[i][1]] = sparseArray[i][2];
        }
        System.out.println("转化后的的二维数组");
        for (int[] row : chessArr1) {
            for (int data : row) {
                //换行输出
                System.out.printf("%d\t",data);
            }
            System.out.println();
        }


    }

}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

​LeetCode解法汇总1572. 矩阵对角线元素的和

目录链接&#xff1a; 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目&#xff1a; https://github.com/September26/java-algorithms 原题链接&#xff1a;力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 描述&#xff1a; 给你一个正…

探秘金和OA:解析任意文件读取漏洞的潜在威胁

是喜是悲&#xff0c;但可以慰藉的是&#xff0c;你总不枉在这世界上活了一场&#xff0c;有了这样的认识&#xff0c;你就会珍重生活&#xff0c;而不会玩世不恭&#xff1b;同时也会给人自身注入一种强大的内在力量…… 漏洞复现 访问url&#xff1a; 构造payload /C6/Jh…

【网络编程(二)】NIO快速入门

NIO Java NIO 三大核心组件 Buffer&#xff08;缓冲区&#xff09;&#xff1a;每个客户端连接都会对应一个Buffer&#xff0c;读写数据通过缓冲区读写。Channel&#xff08;通道&#xff09;&#xff1a;每个channel用于连接Buffer和Selector&#xff0c;通道可以进行双向读…

日常问题——使用Java转将long类型为date类型,日期是1970年

&#x1f61c;作 者&#xff1a;是江迪呀✒️本文关键词&#xff1a;日常BUG、BUG、问题分析☀️每日 一言 &#xff1a;存在错误说明你在进步&#xff01; 一、问题描述 long类型的日期为&#xff1a;1646718195 装换为date类型&#xff1a; Date date new Dat…

SQL | 使用函数处理数据

8-使用函数处理数据 8.1-函数 SQL可以用函数来处理数据。函数一般是在数据上执行的&#xff0c;为数据的转换和处理提供了方便。 8.1.1 函数带来的问题 每种DBMS都有特定的函数&#xff0c;只有很少一部分函数&#xff0c;是被所有主要的DBMS等同的支持。 虽然所有的类型的…

Linux 基础(五)常用命令-文件属性

文件属性 文件权限文件属性修改文件权限属性 文件所有者 文件权限 文件属性 Linux中文件权限 可以通过文件属性体现&#xff1b; 使用 ll 查看文件列表 最前面的 l d 表示文件类型 1 5 表示硬链接数 或者 子文件夹个数 所属用户 所属用户组 文件大小 创建/更新时间 文件&…

【前端二次开发框架关于关闭eslint】

前端二次开发框架关于关闭eslint 方法一方法二方法三方法四&#xff1a;以下是若想要关闭项目中的部分代码时&#xff1a; 方法一 在vue.config.js里面进行配置&#xff1a; module.exports {lintOnSave:false,//是否开启eslint保存检测 ,它的有效值为 true || false || err…

一个简单实用的线程池及线程池组的实现!

1.线程池简介 线程池&#xff0c;顾名思义&#xff0c;就是一个“池子”里面放有多个线程。为什么要使用线程池呢&#xff1f;当我们编写的代码需要并发异步处理很多任务时候&#xff0c;一般的处理办法是一个任务开启一个线程去处理&#xff0c;处理结束后释放线程。可是这样…

Docker安装 Kibana

目录 前言安装Kibana步骤1&#xff1a;准备1. 安装docker2. 搜索可以使用的镜像。3. 也可从docker hub上搜索镜像。4. 选择合适的redis镜像。 步骤2&#xff1a;拉取 kibana 镜像拉取镜像查看已拉取的镜像 步骤3&#xff1a;创建容器创建容器方式1&#xff1a;快速创建容器 步骤…

vue父页面给iframe子页面传值

在vue父页面有两个个参数 名称和图标&#xff0c;需要把这两个参数传到iframe的地图里面&#xff0c;在地图触发绘点事件的时候&#xff0c;获取到传来的参数并且展示 vue:传值给子页面iframe // 传值给子页面iframe(2个参数)handleIframeLoad() {const iframeWindow this.$re…

海康威视iVMS综合安防系统任意文件上传(0Day)

漏洞描述 攻击者通过请求/svm/api/external/report接口任意上传文件,导致获取服务器webshell权限,同时可远程进行恶意代码执行。 免责声明 技术文章仅供参考,任何个人和组织使用网络应当遵守宪法法律,遵守公共秩序,尊重社会公德,不得利用网络从事危害国家安全、荣誉和…

安全第二次

一&#xff0c;iframe <iframe>标签用于在网页里面嵌入其他网页。 1&#xff0c;sandbox属性 如果嵌入的网页是其他网站的页面&#xff0c;因不了解对方会执行什么操作&#xff0c;因此就存在安全风险。为了限制<iframe>的风险&#xff0c;HTML 提供了sandb…

HCIA---动态路由---RIP协议

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 目录 前言 一.动态路由 二.动态路由协议分类 IGP&#xff1a;内部网关协议 EGP:外部网关协议 三.RIP协议概述 RIP版本分类&#xff1a; RIP三要素&#xff1a; 思维…

TypeScript 关于对【泛型】的定义使用解读

目录 概念导读泛型函数多个泛型参数泛型约束泛型别名泛型接口泛型类总结&#xff1a; 概念导读 泛型&#xff08;Generics&#xff09;是指在定义函数、接口或类的时候&#xff0c;不预先指定具体的类型&#xff0c;而在使用的时候再指定类型的一种特性。使用泛型 可以复用类型…

【非科班如何丝滑转码?】探索计算机领域跳槽之路

近年来&#xff0c;计算机领域的蓬勃发展吸引着越来越多非计算机科班出身的人士投身其中。本文将就如何顺利实现非科班转码&#xff0c;计算机岗位的发展前景&#xff0c;以及现阶段转码的建议&#xff0c;结合个人经验和观察&#xff0c;为您阐述详细全面的观点。 一、如何规划…

【雕爷学编程】Arduino动手做(201)---行空板开发环境之VSCode

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

Kotlin 基础教程一

Kotlin 基本数据类型 Java | Kotlin byte Byte short Short int Int long Long float Float double Double boolean Boolean c…

PHP最简单自定义自己的框架view使用引入smarty(8)--自定义的框架完成

1、实现效果。引入smarty&#xff0c; 实现assign和 display 2、下载smarty&#xff0c;创建缓存目录cache和扩展extend 点击下面查看具体下载使用&#xff0c;下载改名后放到extend PHP之Smarty使用以及框架display和assign原理_PHP隔壁老王邻居的博客-CSDN博客 3、当前控…

xmind用例数据上传至禅道

xmind格式参考&#xff0c;只需要在一级子主题填写对应用例模块ID&#xff0c;其余格式随意即可生成用例并直接上传到禅道&#xff1a; 代码里需填写禅道对应登录账号及用例所属产品 import requests import json import re import hashlib import pprint import threading fr…

并发——Atomic 原子类总结

文章目录 1 Atomic 原子类介绍2 基本类型原子类2.1 基本类型原子类介绍2.2 AtomicInteger 常见方法使用2.3 基本数据类型原子类的优势2.4 AtomicInteger 线程安全原理简单分析 3 数组类型原子类3.1 数组类型原子类介绍3.2 AtomicIntegerArray 常见方法使用 4 引用类型原子类4.1…