三角矩阵和对称阵的压缩存储

文章目录

矩阵的存储

  • 对于多维数组,有两种映射方法:按行优先和按列优先。
    A n = ( a 11 a 12 ⋯ a 1 n a 21 a 22 ⋯ a 2 n ⋮ ⋮ ⋱ ⋮ a n 1 a n 2 ⋯ a n n ) A_{n}= \begin{pmatrix} a_{11}& a_{12}& \cdots & a_{1n} \\ a_{21}& a_{22}& \cdots & a_{2n} \\ \vdots & \vdots & \ddots & \vdots \\ a_{n1}& a_{n2}& \cdots & a_{nn} \end{pmatrix} An= a11a21⋮an1a12a22⋮an2⋯⋯⋱⋯a1na2n⋮ann

  • 以二维数组为例

    • 按行优先存储 的基本思想是:先行后列,先存储行号较小的元素,行号相等先存储列号较小的元素。对于上述 A n A_{n} An,存储顺序为 a 11 , a 12 , ⋯   , a n n a_{11},a_{12},\cdots,a_{nn} a11,a12,⋯,ann
      • 设二维数组的行下标与列下标的范围分别为 [ 0 , h 1 , ] , [ 0 , h 2 ] [0,h_{1},],[0,h_2] [0,h1,],[0,h2],则存储结构关系式为 L o c ( a i j ) = L o c ( a 00 ) + ( i ( h 2 + 1 ) + j ) L \rm{Loc(a_{ij})}=Loc(a_{00})+(i(h_{2}+1)+j)L Loc(aij)=Loc(a00)+(i(h2+1)+j)L
    • 按优先存储,存储顺序为 a 11 , a 21 , ⋯   , a n n a_{11},a_{21},\cdots,a_{nn} a11,a21,⋯,ann
      • L o c ( a i j ) = L o c ( a 00 ) + ( j ( h 1 + 1 ) + i ) L \rm{Loc(a_{ij})}=Loc(a_{00})+(j(h_{1}+1)+i)L Loc(aij)=Loc(a00)+(j(h1+1)+i)L
    • 其中 [ i ( h 2 + 1 ) + j ] [i(h_{2}+1)+j] [i(h2+1)+j], [ j ( h 1 + 1 ) + i ] [j(h_{1}+1)+i] [j(h1+1)+i]是偏移数组首地址的元素数量,再乘以单个元素所占存储单元数 L = s i z e o f ( E l e m T y p e ) L=\rm{sizeof(ElemType)} L=sizeof(ElemType),即可得到 L o c ( a i j ) \rm{Loc(a_{ij})} Loc(aij)

对称矩阵

对称矩阵的定义及其存储方式。

对称矩阵是指一个 n n n 阶矩阵 A A A 中的任意一个元素 a i j a_{ij} aij 都有 a i j = a j i a_{ij} = a_{ji} aij=aji( 1 ≤ i , j ≤ n 1 \leq i, j \leq n 1≤i,j≤n),即矩阵与其转置相等。对称矩阵的元素可以划分为三个部分:上三角区、主对角线和下三角区

对于 n n n 阶对称矩阵,上三角区的所有元素和下三角区的对应元素相同。若仍采用二维数组存放,则会浪费几乎一半的空间。因此,可以将 n n n 阶对称矩阵 A A A 存放在一维数组 B [ n ( n + 1 ) / 2 ] B[n(n+1)/2] B[n(n+1)/2] 中,即元素 a i j a_{ij} aij 存放在 b k b_k bk 中。比如只存放下半部分(含主对角线)的元素。

在数组 B B B 中,位于元素 a i j a_{ij} aij( i ≥ j i \geq j i≥j)前面的元素个数为:

  • 第 1 1 1 行: 1 1 1 个元素( a 1 , 1 a_{1,1} a1,1)。
  • 第 2 2 2 行: 2 2 2 个元素( a 2 , 1 , a 2 , 2 a_{2,1}, a_{2,2} a2,1,a2,2)。
  • ......
  • 第 i − 1 i-1 i−1 行: i − 1 i-1 i−1 个元素( a i − 1 , 1 , a i − 1 , 2 , ⋯   , a i − 1 , j − 1 a_{i-1,1}, a_{i-1,2}, \cdots, a_{i-1,j-1} ai−1,1,ai−1,2,⋯,ai−1,j−1)。
  • 第 i i i 行: j − 1 j-1 j−1 个元素( a i , 1 , a i , 2 , ⋯   , a i , j − 1 a_{i,1}, a_{i,2}, \cdots, a_{i,j-1} ai,1,ai,2,⋯,ai,j−1)。

因此,元素 a i j a_{ij} aij 在数组 B B B 中的下标 k = 1 + 2 + ⋯ + ( i − 1 ) + j − 1 k = 1 + 2 + \cdots + (i-1) + j-1 k=1+2+⋯+(i−1)+j−1= i ( i − 1 ) / 2 + j − 1 i(i-1)/2 + j-1 i(i−1)/2+j−1(数组下标从 0 0 0 开始)

即 k ( i , j ) k(i,j) k(i,j)= i ( i − 1 ) / 2 + j − 1 i(i-1)/2 + j-1 i(i−1)/2+j−1, ( i ⩾ j ) (i\geqslant{j}) (i⩾j)(1)

上面讨论的是下三角内 a i j a_{ij} aij, ( i ⩾ j ) (i\geqslant{j}) (i⩾j)条件下的 i , j , k i,j,k i,j,k关系式,给出了 a i j , ( i ⩾ j ) a_{ij},(i\geqslant{j}) aij,(i⩾j)存储到B中的位置索引 k k k,即 B [ k ] = a i j B[k]=a_{ij} B[k]=aij,

在压缩对称阵的时候,可以仅遍历矩阵的下三角形以及主对角线部分的元素;上三角部分的元素不用遍历,就可以存储这个矩阵的元素的完整信息,可以无损解压出来

那么 i < j i<j i<j的情况下 i , j , k i,j,k i,j,k之间的关系又是如何?或者说若 a i j , ( i < j ) a_{ij},(i<j) aij,(i<j)时 a i j a_{ij} aij对应于数组B中的索引为 k k k的元素,那么 k , i , j k,i,j k,i,j之间满足什么关系

注意到对称矩阵的特点: a i , j = a j , i a_{i,j}=a_{j,i} ai,j=aj,i,这样就可以将 i < j i<j i<j的情况转换到 i > j i>j i>j的情况

当 i < j i<j i<j时 j > i j>i j>i,所以 a j i a_{ji} aji满足公式(1)的条件, k ( j , i ) k(j,i) k(j,i)= j ( j − 1 ) / 2 + i − 1 j(j-1)/2+i-1 j(j−1)/2+i−1, ( j > i ) (j>i) (j>i)

综上元素下标 i , j , k i,j,k i,j,k之间的对应关系如下:
k = { i ( i − 1 ) 2 + j − 1 , i ⩾ j j ( j − 1 ) 2 + i − 1 , i < j k = \begin{cases} \frac{i(i-1)}{2} + j-1, & i \geqslant j \\ \frac{j(j-1)}{2} + i-1, & i < j \end{cases} k={2i(i−1)+j−1,2j(j−1)+i−1,i⩾ji<j

对称阵和其他特殊矩阵不同,对称阵上三角按列优先存储和下三角按行优先存储到一维数组B时序列是一样的;

因此对称阵中遇到上三角案列优先存储的问题,可以考虑转换为对应的按行优先存储的情况来解决,并且,无论哪种存储, a i j a_{ij} aij= a j i a_{ji} aji,当 i > j i>j i>j时总是可以考虑转换为 a j i a_{ji} aji来处理

对于对称阵上三角按列优先存储, a i j a_{ij} aij存储在 B [ k ] B[k] B[k],则 k = 1 + 2 + ⋯ + ( j − 1 ) + i − 1 k=1+2+\cdots+(j-1)+i-1 k=1+2+⋯+(j−1)+i−1= 1 2 j ( j − 1 ) + i − 1 \frac{1}{2}j(j-1)+i-1 21j(j−1)+i−1, ( i ⩽ j ) (i\leqslant{j}) (i⩽j), k = 1 2 i ( i − 1 ) + j − 1 k=\frac{1}{2}i(i-1)+j-1 k=21i(i−1)+j−1, ( i > j ) (i>j) (i>j)

例如 a 7 , 2 a_{7,2} a7,2, k = 1 2 × 7 × 6 + 1 k=\frac{1}{2}\times7\times{6}+1 k=21×7×6+1=22

三角阵

和对称阵的压缩类似,以下三角为例,将元素的下三角和住对角线元素压缩到数组,但是上三角部分元素都是相同且无法通过下三角矩阵推出,需要存储到数组中,对于 n n n阶三角阵,压缩需要的一维数组长度为 1 2 n ( n + 1 ) + 1 \frac{1}{2}n(n+1)+1 21n(n+1)+1,比对称真要多1

三角阵的压缩

  • 三角矩阵(方阵) A [ 1... n ] [ 1... n ] A[1...n][1...n] A[1...n][1...n],包括
    • 上三角矩阵(下三角所有元为同一个常量) j ⩽ i j\leqslant i j⩽i
    • 下三角矩阵(上三角所有元素为同一个常量) i ⩽ j i\leqslant j i⩽j
  • 上下三角包含 n ( n + 1 ) 2 个元素 ; 在加上 ( 占据矩阵其余位置的 ) 一个常量 c 上下三角包含\frac{n(n+1)}{2}个元素;在加上(占据矩阵其余位置的)一个常量c 上下三角包含2n(n+1)个元素;在加上(占据矩阵其余位置的)一个常量c
  • 所以需要一个大小 至少 ‾ 为 n ( n + 1 ) 2 + 1 维的数组来保存必要元素 所以需要一个大小\underline{至少}为\frac{n(n+1)}{2}+1维的数组来保存必要元素 所以需要一个大小至少为2n(n+1)+1维的数组来保存必要元素
    • A [ 1 , . . . , n ] [ 1 , . . . , n ] ; B [ 1... 1 2 n ( n + 1 ) + 1 ] A[1,...,n][1,...,n];B[1...\frac{1}{2}{n(n+1)}+1] A[1,...,n][1,...,n];B[1...21n(n+1)+1]

下三角矩阵

按行优先

推到过程和存储下三角对称阵过程几乎一样
k = { i ( i − 1 ) 2 + j − 1 , i ⩾ j 1 2 n ( n + 1 ) , i < j k = \begin{cases} \frac{i(i-1)}{2} + j-1, & i \geqslant j \\ \frac{1}{2}n(n+1), & i < j \end{cases} k={2i(i−1)+j−1,21n(n+1),i⩾ji<j

这里 i < j i<j i<j的情况是上三角中的元素,上三角中的元素都存储在B[0...n(n+1)/2]中的最后一个元素中,索引为 1 2 n ( n + 1 ) \frac{1}{2}{n}(n+1) 21n(n+1)

如果数组B从1开始计数,那么对上述公式加1即可

按列优先

和按行优先存储的上三角矩阵压缩的推导几乎一样,可以先参考该节

j j j 第 j j j列需要进入数组B的元素个数
1 n n n
2 n − 1 n-1 n−1
3 3 3 n − 2 n-2 n−2
... ...
j − 1 j-1 j−1 n − ( j − 2 ) n-(j-2) n−(j−2)
i i i b i ⩽ n − ( i − 1 ) b_{i}\leqslant n-(i-1) bi⩽n−(i−1)

s j − 1 s_{j-1} sj−1= 1 2 ( n + ( n − ( j − 2 ) ) ) ( j − 1 ) \frac{1}{2}(n+(n-(j-2)))(j-1) 21(n+(n−(j−2)))(j−1)= 1 2 ( 2 n − j + 2 ) ( j − 1 ) \frac{1}{2}(2n-j+2)(j-1) 21(2n−j+2)(j−1)

第 j j j列需要入数组的元素数量为 b j = i − j + 1 b_{j}=i-j+1 bj=i−j+1,(由于是下三角, i ⩾ j i\geqslant{j} i⩾j)

k k k= s j − 1 + ( i − j ) s_{j-1}+(i-j) sj−1+(i−j)= 1 2 ( 2 n − j + 2 ) ( j − 1 ) + ( i − j ) \frac{1}{2}(2n-j+2)(j-1)+(i-j) 21(2n−j+2)(j−1)+(i−j),B[0...n(n+1)/2]

k k k= 1 2 ( 2 n − j + 2 ) ( j − 1 ) + ( i − j ) + 1 \frac{1}{2}(2n-j+2)(j-1)+(i-j)+1 21(2n−j+2)(j−1)+(i−j)+1,B[1...n(n+1)/2+1]

上三角矩阵

分别讨论按行优先和按列优先存储下, i , j , k i,j,k i,j,k的关系

按行优先
i i i 第 i i i行需要进入数组B的元素个数 b i b_{i} bi(行 i i i上属于上三角的元素的个数)
1 n n n
2 n − 1 n-1 n−1
3 3 3 n − 2 n-2 n−2
... ...
i − 1 i-1 i−1 n − ( i − 2 ) n-(i-2) n−(i−2)
i i i b i ⩽ n − ( i − 1 ) b_{i}\leqslant n-(i-1) bi⩽n−(i−1)

上面的表格中,第 i i i行实际上不一定填满,利用第 i i i行的公式,带入 i − 1 i-1 i−1得出第 i − 1 i-1 i−1行需要入B数组的元素个数: n − i + 2 n-i+2 n−i+2

累计前 i − 1 i-1 i−1行的元素的入数组B的元素个数(为了讨论方便,后续省略入数组B 这几个字,直接用元素数量称呼),这是个等差数列求和问题, s i − 1 s_{i-1} si−1= 1 2 ( n + ( n − i + 2 ) ) ( i − 1 ) \frac{1}{2}(n+(n-i+2))(i-1) 21(n+(n−i+2))(i−1)= 1 2 ( i − 1 ) ( 2 n − i + 2 ) \frac{1}{2}(i-1)(2n-i+2) 21(i−1)(2n−i+2),利用的求和公式中没有公差的 s n = 1 2 ( a 1 + a n ) n s_{n}=\frac{1}{2}(a_{1}+a_{n})n sn=21(a1+an)n(注意这里项数 n = i − 1 n=i-1 n=i−1,对应的是求和前 i − 1 i-1 i−1行的元素数量)

然后单独计算第 i i i行的元素数量,这时候 b i = j − i + 1 b_{i}=j-i+1 bi=j−i+1,第 i i i行中 a i j a_{ij} aij前面的元素为 j − i j-i j−i;

因此 a i j a_{ij} aij存入数组B[0...n(n+1)/2]后的索引为

  • k k k= s i − 1 + b i − 1 s_{i-1}+b_{i}-1 si−1+bi−1= 1 2 ( i − 1 ) ( 2 n − i + 2 ) + ( j − i ) \frac{1}{2}(i-1)(2n-i+2)+(j-i) 21(i−1)(2n−i+2)+(j−i), ( i ⩽ j ) (i\leqslant{j}) (i⩽j) 主对角线和上三角区域

  • k = n ( n + 1 ) 2 k=\frac{n(n+1)}{2} k=2n(n+1), ( j < i ) (j<i) (j<i),下三角区域

使用条件大括号描述
k 0 ( i , j ) = { ( i − 1 ) ( 2 n − i + 2 ) 2 + j − i ; ( i ⩽ j ) n ( n + 1 ) 2 ; ( j < i ) k_0(i,j)=\begin{cases} \frac{(i-1)(2n-i+2)}{2}+j-i ;(i\leqslant j) \\\frac{n(n+1)}{2};(j<i) \end{cases} k0(i,j)={2(i−1)(2n−i+2)+j−i;(i⩽j)2n(n+1);(j<i)

下三角区域的元素取值都一样,因此总是将值存放到同一个位置,这个位置一般设置在数组B的最后一个位置(数组的长度为 n ( n + 1 ) 2 + 1 \frac{n(n+1)}{2}+1 2n(n+1)+1,数组B从0开始计数(B[0...n(n+1)/2],那么最后一个元素的索引就是 n ( n + 1 ) 2 \frac{n(n+1)}{2} 2n(n+1)

如果是存入B[1...n(n+1)/2+1],公式会发生变换

  • k 1 = S i − 1 + s i k_1=S_{i-1}+s_i k1=Si−1+si= 1 2 ( i − 1 ) ( 2 n − i + 2 ) + ( j − i + 1 ) \frac{1}{2}(i-1)(2n-i+2)+(j-i+1) 21(i−1)(2n−i+2)+(j−i+1), ( i ⩽ j ) (i\leqslant{j}) (i⩽j)

  • k 1 k_{1} k1= n ( n + 1 ) 2 + 1 \frac{n(n+1)}{2}+1 2n(n+1)+1, ( j < i ) (j<i) (j<i)

按列优先

案列优先存储上三角矩阵的上三角部分和对角线元素,推导过程和按行存储类似

j j j 第 j j j列需要进入数组B的元素个数
1 1 1 1
2 2 2 2
3 3 3 3 3 3
... ...
j − 1 j-1 j−1 j − 1 j-1 j−1
j j j ⩽ j \leqslant j ⩽j

先计算前 j − 1 j-1 j−1列需要压缩的元素数量, s j − 1 s_{j-1} sj−1= 1 2 ( 1 + ( j − 1 ) ) ( j − 1 ) \frac{1}{2}(1+(j-1))(j-1) 21(1+(j−1))(j−1)= 1 2 j ( j − 1 ) \frac{1}{2}j(j-1) 21j(j−1)

第 j j j列的元素数量为 i i i;

所以 k ( i , j ) k(i,j) k(i,j)= 1 2 j ( j − 1 ) + i − 1 \frac{1}{2}j(j-1)+i-1 21j(j−1)+i−1,(B[0...n(n+1)/2])

若存入B[1...n(n+1)/2+1],则 k ( i , j ) k(i,j) k(i,j)= 1 2 j ( j − 1 ) + i \frac{1}{2}j(j-1)+i 21j(j−1)+i

A[0...][0...] vs A[1...][1...]

  • 前面的公式都是在 i ∈ [ 1 , n ] , j ∈ [ 1 , n ] i\in[1,n],j\in[1,n] i∈[1,n],j∈[1,n]情况下推导的
  • 有时矩阵的坐标是从0开始的: i ∈ [ 0 , n ] , j ∈ [ 0 , n ] i\in[0,n],j\in[0,n] i∈[0,n],j∈[0,n]
    • 如果矩阵是A[0...][0...],第一个元素是 a 00 a_{00} a00,那么相关公式不能直接使用
    • 幸运的是,这个调整比较简单,只需要将 i , j i,j i,j直接映射 i + 1 , j + 1 i+1,j+1 i+1,j+1就可以了
相关推荐
小孟Java攻城狮3 小时前
leetcode-不同路径问题
算法·leetcode·职场和发展
查理零世4 小时前
算法竞赛之差分进阶——等差数列差分 python
python·算法·差分
小猿_006 小时前
C语言程序设计十大排序—插入排序
c语言·算法·排序算法
肖田变强不变秃6 小时前
C++实现矩阵Matrix类 实现基本运算
开发语言·c++·matlab·矩阵·有限元·ansys
洛水微寒6 小时前
多张图片读入后组成一个矩阵。怎么读取图片,可以让其读入的形式是:ndarray(a,b,c)分别的含义:a为多少张图片,b*c为图片大小
线性代数·矩阵
熊文豪8 小时前
深入解析人工智能中的协同过滤算法及其在推荐系统中的应用与优化
人工智能·算法
siy233311 小时前
[c语言日寄]结构体的使用及其拓展
c语言·开发语言·笔记·学习·算法
吴秋霖11 小时前
最新百应abogus纯算还原流程分析
算法·abogus
灶龙12 小时前
浅谈 PID 控制算法
c++·算法
菜还不练就废了12 小时前
蓝桥杯算法日常|c\c++常用竞赛函数总结备用
c++·算法·蓝桥杯