数据结构学习1-稀疏数组

【转载请注明出处:From李诗雨—https://blog.csdn.net/cjm2484836553/article/details/91345761

不诗意的女程序猿不是好厨师~

1.一些基础点

1.1数据结构:包括线性结构和非线性结构。

1.2线性结构

  • 特点是数据元素之间存在一对一的线性关系.
  • 线性结构有两种不同的存储结构,即顺序存储结构和链式存储结构。
    顺序存储的线性表称为顺序表,顺序表中的存储元素是连续的。
    链式存储的线性表称为链表,链表中的存储元素不一定是连续的,元素节点中存放数据元素以及相邻元素的地址信息。
  • 线性结构常见的有:数组、队列、链表和栈。

1.3非线性结构

  • 非线性结构包括:二维数组,多维数组,广义表,树结构,图结构

2.稀疏数组

2.1实际使用场景

在这里插入图片描述

举个栗子来说:
我们都玩过五子棋游戏,那么如果让我们自己撸起袖子来写一个,你会怎么做呢?用什么来表示棋盘棋子?黑棋白棋如何表示?如果下到一半累了不想玩了需要保存棋局该怎么做?打了局王者乏了又想来继续下五子棋,又该如何复局呢?


数组1

现在我们来分析一下,我们可以用一个11*11的二维数组来模拟棋局,二维数组的值来代表棋子,如果是1就表示是黑子,如果是2就表示是蓝子。
如果要实现保存棋盘的功能,就要把这个二维数组保存起来。
但是我们看现在这个二维数组有很多的0值,直接保存起来感觉很浪费。那有没有什么办法可以简化这个二维数组呢?


数组2

我们可以把它优化成如上图所示的二维数组,
第一行用来表示原有数组有几行几列吗,多少个不同的值。
从第二行开始每一行表示 行、列的位置 和对应的值。

所以数组2可以得到的信息是:
原有数组有11行11列有2个不同的值,其中在(1,2)位置的值是1,(2,3)位置的值是2。
看,其实就是对原有数组的另一种表示。而且我们把1111的规模编程了33的规模,是不是大大的做到了简化。

好的,不知不觉中我们就引入了稀疏数组,其实啊,简化后的数组就是一个稀疏数组。下面我们来看看什么是稀疏数组。

2.1稀疏数组的小结

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

稀疏数组的处理方法是:
记录数组一共有几行几列,有多少个不同的值。
把具有不同值的元素的行列及值记录在一个小规模的数组中,从而缩小程序的规模。

3.学以致用,撸代码

在这里插入图片描述

现在让我们按下面的要求来撸一段代码,练习一下:

3.0利用稀疏数组完成棋局的保存棋局的恢复功能。

分析:
先将如图所示的棋局,用1111的二维数组表示出来,然后把它转化为对应的稀疏数组,存储到本地文件中,这样就完成“棋局的保存”功能。
我们将文件中的内容读取出来,转化为稀疏数组,然后在转化为对应的11
11的二维数组,这样就完成了“复盘”功能。

光说不练假把式,具体代码如下:

3.1用11*11的二维数组展示棋局。

/**
         * 创建一个11*11的二维数组 ,来模拟五子棋棋盘
         * 其中值1代表黑棋,棋2代表蓝棋
         */
        int[][] gobangArray = new int[11][11];
        gobangArray[1][2] = 1;
        gobangArray[2][3] = 2;

        System.out.println("原始棋局:");
        for (int[] row : gobangArray) {
            for (int element : row) {
                System.out.printf("%d\t", element);
            }
            System.out.println();
        }

打印结果如下:


在这里插入图片描述

3.2 将 原始数组 转化为 稀疏数组

具体思路:
a. 遍历 原始的二维数组,得到有效数据的个数 sum
b. 根据sum 创建 稀疏数组 sparseArray int[sum + 1] [3]
c. 将二维数组的有效数据数据存入到 稀疏数组

  /**
         * 遍历原始棋局,得到棋子总数
         */
        int sum = 0;
        for (int[] row : gobangArray) {
            for (int element : row) {
                if (element != 0) sum++;
            }
        }

        /**
         * 创建稀疏数组 int[sum+1][3]
         * 第一行存原始棋局的行数,列数 和 棋子数
         * 后面的每一行:第一列 存棋子的所在行下标,第二列 存棋子的所在列下标,第三列 存值
         */
        int[][] sparseArray = new int[sum + 1][3];
        sparseArray[0][0] = 11;
        sparseArray[0][1] = 11;
        sparseArray[0][2] = sum;
        int sparseRow = 1;
        for (int row = 0; row < 11; row++) {
            for (int column = 0; column < 11; column++) {
                if (gobangArray[row][column] != 0) {
                    sparseArray[sparseRow][0] = row;
                    sparseArray[sparseRow][1] = column;
                    sparseArray[sparseRow][2] = gobangArray[row][column];
                    sparseRow++;
                }
            }
        }

        /**
         * 验证一下
         */
        System.out.println();
        System.out.println("稀疏数组为:");
        for (int[] row : sparseArray) {
            for (int element : row) {
                System.out.printf("%d\t", element);
            }
            System.out.println();
        }

打印结果如下:


在这里插入图片描述

3.3 将稀疏数组保存到本地的txt文件中

 //将稀疏数组存入到文件中
        File file = new File("e:\\array.txt");  //存放数组数据的文件
        try {
            FileWriter out = new FileWriter(file);  //文件写入流
            //将数组中的数据写入到文件中。每行各数据之间用TAB间隔
            for(int i=0;i<sparseArray.length;i++){
                for(int j=0;j<3;j++){
                    out.write(sparseArray[i][j]+"\t");
                }
                out.write("\r\n");
            }
            out.close();
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("存入文件时出现异常:"+e.getMessage().toString());
        }

此时在本地的e盘下我们会发现多了一个array.txt文件,打开这个文件,发现存储正确。

在这里插入图片描述

哈哈,到此模拟 棋局的保存 功能就完成了。

3.4 从本地txt文件中读取数据,并转化为对应的稀疏数组。

 /**
         * 从txt文件中读取数据
         */
        File txtFile = new File("e:\\array.txt");
        int[][] txtSparseArr = new int[3][3];
        try {
            String line;//存放每行的数据
            int row = 0;//记录行数
            if (txtFile.exists()) {
                BufferedReader in = new BufferedReader(new FileReader(txtFile));
                while ((line = in.readLine()) != null) {
                    String[] temp = line.split("\t");
                    for (int j = 0; j < temp.length; j++) {
                        txtSparseArr[row][j] = Integer.parseInt(temp[j]);
                    }
                    row++;
                }
            } else {
                System.out.println("没有找到文件");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        /**
         * 验证一下
         */
        System.out.println();
        System.out.println("txt中得到的稀疏数组:");
        for (int[] lineData : txtSparseArr) {
            for (int ele : lineData) {
                System.out.printf("%d\t", ele);
            }
            System.out.println();
        }

打印结果如下:


在这里插入图片描述

3.5 将稀疏数组恢复为原始数组

具体思路:
a. 先读取稀疏数组的第一行,根据第一行的数据,创建原始的二维数组,比如上面的 int[][] gobangArray2 = new int[11][11]。
b. 在读取稀疏数组后几行的数据,并赋给 原始的二维数组 即可。

 /**
         * 根据稀疏数组恢复棋局
         * 遍历稀疏数组的每行,给新棋局进行赋值
         */
        int row1 = txtSparseArr[0][0];
        int column1 = txtSparseArr[0][1];
        int[][] gobangArray2 = new int[row1][column1];
        for (int i = 1; i < txtSparseArr.length; i++) {//需从第二行开始
            int row2 = txtSparseArr[i][0];
            int column2 = txtSparseArr[i][1];
            int value = txtSparseArr[i][2];
            gobangArray2[row2][column2] = value;
        }

        System.out.println();
        System.out.println("复局后的新棋盘");
        for (int[] row : gobangArray2) {
            for (int element : row) {
                System.out.printf("%d\t", element);
            }
            System.out.println();
        }

打印结果如下:


在这里插入图片描述

这样 “恢复棋局” 功能也实现了。

思考:如果不知道txt中二维数组的行、列数呢?该如何处理?

我的想法是把txt中的每行数据都装进一个ArrayList中,这样ArrayList.size就是行数,每个行的内容进行拆分后形成的数组的长度就是列数。这样就可以创建稀疏数组了,然后遍历,把数据放入稀疏数组中。这样也可以把txt文件转换为二维数组了。
这个只是自己笨拙的思考,如果大佬有好的方法,希望可以告诉我一下哈。

 File file5 = new File("e:\\array.txt");
        ArrayList<String> txtData = new ArrayList<>();
        try {
            String line;
            if (file5.exists()) {
                BufferedReader in = new BufferedReader(new FileReader(file5));
                while ((line = in.readLine()) != null) {
                    txtData.add(line);
                }
            } else {
                System.out.println("没有找到文件");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        if (txtData != null && txtData.size() > 0) {
            String[] firstLine = txtData.get(0).split("\t");
            int rows = txtData.size();//得到行数
            int columns = firstLine.length;//得到列数
            int[][] sparseArray3 = new int[rows][columns];
            for (int i = 0; i < txtData.size(); i++) {
                String[] oneLine = txtData.get(i).split("\t");
                //如果数组 sparseArray3 和 onLine 中的元素类型相同,则只要写此行就可以了
//                sparseArray3[i] = oneLine;
                for (int j = 0; j < oneLine.length; j++) {
                    sparseArray3[i][j] = Integer.parseInt(oneLine[j]);
                }
            }

            System.out.println();
            System.out.println("新方式得到的稀疏数组222:");
            for (int[] lineData : sparseArray3) {
                for (int ele : lineData) {
                    System.out.printf("%d\t", ele);
                }
                System.out.println();
            }
        }

打印结果如下:


在这里插入图片描述
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容

  • 数组 从本质上讲,数组与顺序表、链表、栈和队列一样,都用来存储具有 "一对一" 逻辑关系数据的线性存储结构。只因各...
    hadoop_a9bb阅读 4,287评论 0 8
  • 一些概念 数据结构就是研究数据的逻辑结构和物理结构以及它们之间相互关系,并对这种结构定义相应的运算,而且确保经过这...
    Winterfell_Z阅读 5,710评论 0 13
  • 这是16年5月份编辑的一份比较杂乱适合自己观看的学习记录文档,今天18年5月份再次想写文章,发现简书还为我保存起的...
    Jenaral阅读 2,739评论 2 9
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,373评论 0 17
  • 读舒己怀《散文//心安处》有感 湖畔柳条抚岸。 亭却只闻悠曲? 伊人霁影回。 往事几多扬去。 空语,空语。 屡梦魂...
    青衫湿旧阅读 288评论 17 19