数组 – 稀疏数组

一,稀疏数组1.定义稀疏数组可以看做是普通数组的压缩,但是这里说的普通数组是值无效数据量远大于有效数据量的数组形如:00000000000001000000000000200000000000000…

大家好,又见面了,我是你们的朋友全栈君。

一,稀疏数组

1.定义

稀疏数组可以看做是普通数组的压缩,但是这里说的普通数组是值无效数据量远大于有效数据量的数组

形如:

              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 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

其稀疏数组形式:

              11 11 2
              1  2  1
              2  4  2

2.存储

刚说到稀疏数组是一种压缩后的数组,为什么要进行压缩存储呢?

  • 原数组中存在大量的无效数据,占据了大量的存储空间,真正有用的数据却少之又少

  • 压缩存储可以节省存储空间以避免资源的不必要的浪费,在数据序列化到磁盘时,压缩存储可以提高IO效率

3.存储方式

1.普通存储

第一行存储原始数据总行数,总列数,总的非0数据个数

接下来每一行都存储非0数所在行,所在列,和具体值

形如:

    rows cols n
    r1   c1   val1
    r2   c2   val2
    .    .     .
    .    .     .
    rn   cn   valn

eg :

              11 11 2
              1  2  1
              2  4  2

2.链式存储

以这个为例:

        0 0 0 0 
        0 1 0 0
        0 0 2 0
        0 0 3 0

a.普通链式存储

# 一,稀疏数组

b.行式链式存储

在这里插入图片描述

c.十字链式存储

在这里插入图片描述

4.代码实现

/** * @Description: 稀疏数组 ---> 数组中有很多无效数据,压缩数组 * @Author: Kevin * @CreateDate: 2019/7/2 22:39 * @UpdateUser: Kevin * @UpdateDate: 2019/7/2 22:39 * @UpdateRemark: 修改内容 * @Version: 1.0 */
public class SparseArray { 
   

    /** * <p> * 稀疏数组可以简单的看作为是压缩,在开发中也会使用到。比如将数据序列化到磁盘上,减少数据量,在IO过程中提高效率等等。 * * <p> * 为什么要进行压缩? * - 由于稀疏矩阵中存在大量的“空”值,占据了大量的存储空间,而真正有用的数据却少之又少, * - 且在计算时浪费资源,所以要进行压缩存储以节省存储空间和计算方便。 * </p> * * </p> * @param args */

    public static void main(String[] args) { 
   

        /** * 初始化二维数组 * <p> * 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 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 * </p> */
        //初始化原数组
        int[][] array = new int[11][11];
        array[1][2] = 1;
        array[2][4] = 2;
        for(int[] row : array){ 
   
            for(int item : row){ 
   
                System.out.printf("%d\t",item);
            }
        }

        System.out.println("---------> 二维数组转稀疏数组");

        /** * 稀疏数组 * <p> * 11 11 2 * 1 2 1 * 2 4 2 * </p> */
        //得到非0数据数
        int sum = 0;
        for (int i = 0;i<11;i++){ 
   
            for(int j = 0;j<11;j++){ 
   
                if(array[i][j] != 0){ 
   
                    sum++;
                }
            }
        }
        //创建稀疏数组
        int[][] sparseArray = new int[sum+1][3];
        //给稀疏数组赋值
        sparseArray[0][0] = 11;
        sparseArray[0][1] = 11;
        sparseArray[0][2] = sum;
        //将非0的数放入稀疏数组
        //count:标识第几个非0数
        int count = 0;
        for (int i = 0;i<11;i++){ 
   
            for(int j = 0;j<11;j++){ 
   
                if(array[i][j] != 0){ 
   
                    count++;
                    sparseArray[count][0] = i;
                    sparseArray[count][1] = j;
                    sparseArray[count][2] = array[i][j];
                }
            }
        }
        //遍历稀疏数组
        for(int i = 0;i<sparseArray.length;i++){ 
   
            System.out.printf("%d%d%d\t",sparseArray[i][0],sparseArray[i][1],sparseArray[i][2]);
        }

        System.out.println("----------->稀疏数组转回原始数组");

        /** * 恢复的二维数组 * <p> * 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 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 * </p> */

        int[][] oldArray = new int[sparseArray[0][0]][sparseArray[0][1]];
        //将原来非0的数填充回去
        for(int i = 1;i<=count;i++){ 
   
          oldArray[sparseArray[i][0]][sparseArray[i][1]] = sparseArray[i][2];
        }
        //遍历刚转回的原始数组
        for(int[] row : oldArray){ 
   
            for(int item : row){ 
   
                System.out.printf("%d\t",item);
            }
        }
    }
}

3.将稀疏数组存到此磁盘中

我们可以使用java的IO流将稀疏数组存放到磁盘中,原数组和稀疏数组比较,肯定是稀疏数组体积更小,占用空间更小

        /** * 将稀疏数组存入磁盘(文件) * */
        public static void sparseArrayToIo(int[][] sparseArray) throws Exception { 
   
            File file = new File("sparseArray.txt");
            if(!file.exists()){ 
   
                file.createNewFile();
            }
            FileWriter writer = new FileWriter(file);
            for(int i =0; i < sparseArray.length; i++) { 
   
                for(int j = 0; j < 3; j++) { 
   
                    writer.write(sparseArray[i][j]);
                }
            }
            writer.flush();
            writer.close();
        }

4.从磁盘中读取稀疏数组

在这里有个缺陷就是我不能动态的知道稀疏数组一共有几行,所以我选择传参的方式,这样其实是不太友好的

        /** * 读文件获取稀疏数组(获取指定行数的稀疏数组)【不足】 * @return */
        public static int[][] sparseArrayFromIo(int lineNums) throws Exception { 
   

            FileReader reader = new FileReader("sparseArray.txt");
            int getNum = 0;
            int[][] sparseArray = new int[lineNums][3];
            for(int i = 0;i < lineNums;i++) { 
   
                for (int j = 0; j < 3; j++) { 
   
                    getNum = reader.read();
                    sparseArray[i][j] = getNum;
                }
            }
            return sparseArray;
        }

恢复原数组

            System.out.println("----------->稀疏数组转回原始数组");
            //读取磁盘中的稀疏数组
            try { 
   
                int[][] sparseArrayFromIo = sparseArrayFromIo(3);
                int[][] newOldArray = new int[sparseArrayFromIo[0][0]][sparseArrayFromIo[0][1]];
                //将原来非0的数填充回去
                for(int i = 1;i<=count;i++){ 
   
                    newOldArray[sparseArrayFromIo[i][0]][sparseArrayFromIo[i][1]] = sparseArrayFromIo[i][2];
                }
                //遍历刚转回的原始数组
                for(int[] row : newOldArray){ 
   
                    for(int item : row){ 
   
                        System.out.printf("%d\t",item);
                    }
                }
            } catch (Exception e) { 
   
                e.printStackTrace();
            }

5.完整代码

public class SparseArray { 
   

        /** * <p> * 稀疏数组可以简单的看作为是压缩,在开发中也会使用到。比如将数据序列化到磁盘上,减少数据量,在IO过程中提高效率等等。 * * <h> * 为什么要进行压缩? * - 由于稀疏矩阵中存在大量的“空”值,占据了大量的存储空间,而真正有用的数据却少之又少, * - 且在计算时浪费资源,所以要进行压缩存储以节省存储空间和计算方便。 * </h> * * </p> * @param args */

        public static void main(String[] args) { 
   

            /** * 初始化二维数组 * <p> * 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 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 * </p> */
            //初始化原数组
            int[][] array = new int[11][11];
            array[1][2] = 1;
            array[2][4] = 2;
            for(int[] row : array){ 
   
                for(int item : row){ 
   
                    System.out.printf("%d\t",item);
                }
            }

            System.out.println("---------> 二维数组转稀疏数组");

            /** * 稀疏数组 * <p> * 11 11 2 * 1 2 1 * 2 4 2 * </p> */
            //得到非0数据数
            int sum = 0;
            for (int i = 0;i<11;i++){ 
   
                for(int j = 0;j<11;j++){ 
   
                    if(array[i][j] != 0){ 
   
                        sum++;
                    }
                }
            }
            //创建稀疏数组
            int[][] sparseArray = new int[sum+1][3];
            //给稀疏数组赋值
            sparseArray[0][0] = 11;
            sparseArray[0][1] = 11;
            sparseArray[0][2] = sum;
            //将非0的数放入稀疏数组
            //count:标识第几个非0数
            int count = 0;
            for (int i = 0;i<11;i++){ 
   
                for(int j = 0;j<11;j++){ 
   
                    if(array[i][j] != 0){ 
   
                        count++;
                        sparseArray[count][0] = i;
                        sparseArray[count][1] = j;
                        sparseArray[count][2] = array[i][j];
                    }
                }
            }
            //遍历稀疏数组
            for(int i = 0;i<sparseArray.length;i++){ 
   
                System.out.printf("%d%d%d\t",sparseArray[i][0],sparseArray[i][1],sparseArray[i][2]);
            }
            try { 
   
                //将稀疏数组写入文件
                sparseArrayToIo(sparseArray);
            } catch (Exception e) { 
   
                e.printStackTrace();
            }
            System.out.println("----------->稀疏数组转回原始数组");
            /** * 恢复的二维数组 * <p> * 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 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 * </p> */
            int[][] oldArray = new int[sparseArray[0][0]][sparseArray[0][1]];
            //将原来非0的数填充回去
            for(int i = 1;i<=count;i++){ 
   
              oldArray[sparseArray[i][0]][sparseArray[i][1]] = sparseArray[i][2];
            }
            //遍历刚转回的原始数组
            for(int[] row : oldArray){ 
   
                for(int item : row){ 
   
                    System.out.printf("%d\t",item);
                }
            }
            System.out.println("----------->稀疏数组转回原始数组");
            //读取磁盘中的稀疏数组
            try { 
   
                int[][] sparseArrayFromIo = sparseArrayFromIo(3);
                int[][] newOldArray = new int[sparseArrayFromIo[0][0]][sparseArrayFromIo[0][1]];
                //将原来非0的数填充回去
                for(int i = 1;i<=count;i++){ 
   
                    newOldArray[sparseArrayFromIo[i][0]][sparseArrayFromIo[i][1]] = sparseArrayFromIo[i][2];
                }
                //遍历刚转回的原始数组
                for(int[] row : newOldArray){ 
   
                    for(int item : row){ 
   
                        System.out.printf("%d\t",item);
                    }
                }
            } catch (Exception e) { 
   
                e.printStackTrace();
            }
        }

        /** * 将稀疏数组存入磁盘(文件) * */
        public static void sparseArrayToIo(int[][] sparseArray) throws Exception { 
   
            File file = new File("sparseArray.txt");
            if(!file.exists()){ 
   
                file.createNewFile();
            }
            FileWriter writer = new FileWriter(file);
            for(int i =0; i < sparseArray.length; i++) { 
   
                for(int j = 0; j < 3; j++) { 
   
                    writer.write(sparseArray[i][j]);
                }
            }
            writer.flush();
        }

        /** * 读文件获取稀疏数组(获取指定行数的稀疏数组)【不足】 * @return */
        public static int[][] sparseArrayFromIo(int lineNums) throws Exception { 
   

            FileReader reader = new FileReader("sparseArray.txt");
            int getNum = 0;
            int[][] sparseArray = new int[lineNums][3];
            for(int i = 0;i < lineNums;i++) { 
   
                for (int j = 0; j < 3; j++) { 
   
                    getNum = reader.read();
                    sparseArray[i][j] = getNum;
                }
            }
            return sparseArray;
        }

}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/125468.html原文链接:https://javaforall.net

(0)
上一篇 2022年4月5日 下午4:35
下一篇 2022年4月5日 下午4:35


相关推荐

  • 茜在人名可以读xi吗_茜读xi还是qian?

    茜在人名可以读xi吗_茜读xi还是qian?01茜可以读xi,也可以读qian,通常用于外国女子名字译音时多读xī,而qiàn多用于中国名。茜,本义是草名即“茜草”读音为qiàn时,指茜草,在古汉语中也指“深红”。读xī时,多为音译用字,多用于人名。茜汉语常用字,读作qiàn或者xī,最早见于《说文》小篆。本义是指一种草名,即茜草,为多年生草本植物,可作红色染料,后引申为大红色、秀美、生动等。茜草【qiàncǎo】,多年生草本植物,根圆锥…

    2022年6月21日
    79
  • Linux内存映射——mmap

    Linux内存映射——mmap一mmap系统调用1.内存映射所谓的内存映射就是把物理内存映射到进程的地址空间之内,这些应用程序就可以直接使用输入输出的地址空间,从而提高读写的效率。Linux提供了mmap()函数,用来映射物理内存。在驱动程序中,应用程序以设备文件为对象,调用mmap()函数,内核进行内存映射的准备工作,生成vm_area_struct结构体,然后调用设备驱动程序中定义的mmap函数。2.

    2022年6月16日
    51
  • Pycharm轻松创建Flask项目

    Pycharm轻松创建Flask项目打开Pycharm的file,选择创建新的项目,然后弹出对话框,我们可以看到里面有很多的案例,Flask、Django等等,我们选择生成Flask的demo程序。选择创建之后一个简易的Flask项目就出现在我们眼前,第一个是入口程序,还有一个static的静态目录,templates是模板存放的位置。在Pycharm上面有个run,我们可以选择run来启动Flask的服务,默认打开…

    2022年8月25日
    5
  • phpstorm 2021激活码_在线激活

    (phpstorm 2021激活码)最近有小伙伴私信我,问我这边有没有免费的intellijIdea的激活码,然后我将全栈君台教程分享给他了。激活成功之后他一直表示感谢,哈哈~IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/100143.html…

    2022年3月26日
    62
  • Pprof

    PprofPprof

    2022年8月1日
    8
  • GLM-4-Flash是什么?一文让你看懂GLM-4-Flash的技术原理、主要功能、应用场景

    GLM-4-Flash是什么?一文让你看懂GLM-4-Flash的技术原理、主要功能、应用场景

    2026年3月12日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号