数据结构——特殊矩阵的压缩存储

数据结构——特殊矩阵的压缩存储

矩阵的压缩存储:将矩阵的元素按照某种分布规律存储在较小的存储单元中。

1、对称矩阵的压缩存储

          n阶对称矩阵:一个n阶的矩阵A中的元素满足a(ij)=a(ji)(0<i,j<n-1)。由于矩阵中的元素是关于主对角线对称的,那么就可以只存储上三角或者下三角矩阵中的元素,这样就可以把原来需要存储n*n个元素压缩至n*(n-1)/2个元素了。使用一维数组s[k]以行为主存储对称矩阵A(i,j)的下三角元素,那么k和(i,j)之间的对应关系如下:

(使用等差数列公式就可以推导出来,下同)

  
  
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <malloc.h>
  4. #include <string.h>
  5. /**对称矩阵压缩存储**/
  6. void PrintMatrix(double *s,int n);
  7. /**存储对称矩阵**/
  8. void storeMatrix(int n){
  9. double *s = new double[n];
  10. int len = n*(n+ 1)/ 2;
  11. printf( "Input the UP_Matrix's element by the order of row\n");
  12. for( int i = 0;i<len;i++){
  13. scanf( "%d",&s[i]);
  14. }
  15. PrintMatrix(s,n);
  16. }
  17. /**打印矩阵**/
  18. void PrintMatrix(double *s,int n){
  19. int k = 0;
  20. for( int i = 0;i<n;i++){
  21. for( int j = 0;j<n;j++){
  22. //以行为主存储下三角矩阵
  23. if(i>=j) k = i*(i+ 1)/ 2+j;
  24. else k = j*(j+ 1)/ 2+i;
  25. printf( "%d\t",s[k]);
  26. }
  27. printf( "\n");
  28. }
  29. }
  30. void main(){
  31. int n = 0 ;
  32. printf( "Input the order of the matrix:");
  33. scanf( "%d",&n);
  34. storeMatrix(n);
  35. system( "pause");
  36. }

2、三角矩阵的压缩存储

      上三角矩阵:下三角元素均为常数的矩阵。
      下三角矩阵:上三角元素均为常数的矩阵。
       跟对称矩阵存储相似,只是一位数组s的最后一位用来存储常数项。而且这里的上三角矩阵的存储相当于对称矩阵的以列为主的压缩存储。

3、对角矩阵的压缩存储

      对角矩阵:所有非零元素集中在主对角线两侧的带状区域内。m对角矩阵是指非零元素在每行中有m个,如下图为三对角矩阵。

      特点:
  • 第一行和最后一行有2个非零元素,第2-n-1行有3个非零元素
  • 前i-1行有3*(i-1)-1个非零元素,第i行的非零元素个数为j-i+1当j<i时,j-i = -1当j>i时j-i=1
  • 使用一维数组存储,需要存储3*n-2个元素
  • 一维数组s[k]和A[i][j]的对应关系为:k = 2*i+j

4、稀疏矩阵的压缩存储

      稀疏矩阵:大多数元素均为零的矩阵。定义矩阵m*n如果有t个非零元素,那么s = t/m*n称为矩阵的稀疏因子,如果s<=0.05那么矩阵为稀疏矩阵。
       数据对象集合:稀疏矩阵可以使用三元组顺序表表示,其中三元组格式为(i,j,e)记录了非零元素的行号、列号以及非零元素。

      基本操作集合:
  • CreateMatrix(&M):创建稀疏矩阵
  • DestroyMatrix(&M):销毁稀疏矩阵
  • PrintMatrix(M):打印稀疏矩阵
  • CopyMatrix(M,&N):复制稀疏矩阵M到N
  • AddMatrix(M,N,&Q):矩阵加法
  • SubMatrix(M,N,&Q):矩阵减法
  • MultiMatrix(M,N,&Q):矩阵乘法
  • TransposeMatrix(M,&N):矩阵转置

代码实现:


 
 
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <malloc.h>
  4. #include <string.h>
  5. /**稀疏矩阵压缩存储**/
  6. #define MAXSIZE 100
  7. typedef struct
  8. {
  9. int i;
  10. int j;
  11. int e;
  12. }Triple;
  13. typedef struct
  14. {
  15. Triple data[MAXSIZE];
  16. int m,n,len;
  17. }Matrix;
  18. //函数声明
  19. int CreateMatrix(Matrix *M);
  20. void DestroyMatrix(Matrix *M);
  21. void PrintMatrix(Matrix M);
  22. void CopyMatrix(Matrix M,Matrix *N);
  23. int CompareNumber(int a,int b);
  24. int AddMatrix(Matrix M,Matrix N,Matrix *Q);
  25. int SubMatrix(Matrix M,Matrix N,Matrix *Q);
  26. void TransposeMatrix(Matrix M,Matrix *N);
  27. /**创建一个稀疏矩阵**/
  28. int CreateMatrix(Matrix *M){
  29. //创建成功返回1
  30. int i,m,n,len;
  31. int e;
  32. int flag;
  33. printf( "Input the Matrix's row,col and the number of non-zero element:");
  34. scanf( "%d%d%d",&m,&n,&len);
  35. if(len>MAXSIZE)
  36. return 0;
  37. M->len = len;
  38. M->m = m;
  39. M->n = n;
  40. //以行为主录入非零元素
  41. for(i = 0;i<len;i++){
  42. do{
  43. printf( "%d、Input the row(1-%d) and col(i-%d) of element by the order of row:",i,M->m,M->n);
  44. scanf( "%d%d%d",&m,&n,&e);
  45. flag = 0;
  46. if(m< 0||n< 0||m>M->m||n>M->n)
  47. flag = 1;
  48. if(i> 0&&m<M->data[i -1].i||m==M->data[i -1].i&&n<M->data[i -1].j)
  49. flag = 1;
  50. } while(flag);
  51. M->data[i].i = m;
  52. M->data[i].j = n;
  53. M->data[i].e = e;
  54. }
  55. return 1;
  56. }
  57. /**销毁一个稀疏矩阵**/
  58. void DestroyMatrix(Matrix *M){
  59. M->len = 0;
  60. M->m = 0;
  61. M->n = 0;
  62. }
  63. /**打印矩阵**/
  64. void PrintMatrix(Matrix M){
  65. int len = M.len;
  66. printf( "k\ti\tj\te\n");
  67. for( int i = 0;i<len;i++){
  68. printf( "%d\t%d\t%d\t%d\n",i,M.data[i].i,M.data[i].j,M.data[i].e);
  69. }
  70. }
  71. /**矩阵复制**/
  72. void CopyMatrix(Matrix M,Matrix *N){
  73. int len = M.len;
  74. N->len = len;
  75. N->m = M.m;
  76. N->n = M.n;
  77. for( int i = 0 ;i<len;i++){
  78. N->data[i].i = M.data[i].i;
  79. N->data[i].j = M.data[i].j;
  80. N->data[i].e = M.data[i].e;
  81. }
  82. }
  83. int CompareNumber(int a,int b){
  84. if(a<b)
  85. return -1;
  86. if(a==b)
  87. return 0;
  88. return 1;
  89. }
  90. /**矩阵加法**/
  91. int AddMatrix(Matrix M,Matrix N,Matrix *Q){
  92. //相加成功返回1
  93. if(M.m!=N.m||M.n!=N.n) return 0;
  94. int m = 0,n = 0,k = -1;
  95. Q->m = M.m;
  96. Q->n = M.n;
  97. while(m<M.len&&n<M.len){
  98. switch (CompareNumber(M.data[m].i,N.data[n].i))
  99. {
  100. case -1:
  101. Q->data[++k] = M.data[m++];
  102. break;
  103. case 0:
  104. switch (CompareNumber(M.data[m].j,N.data[n].j))
  105. {
  106. case -1:
  107. Q->data[++k] = M.data[m++];
  108. break;
  109. case 0:
  110. Q->data[++k] = M.data[m++];
  111. Q->data[k].e += N.data[n++].e;
  112. if(Q->data[k].e== 0){
  113. k--;
  114. }
  115. break;
  116. case 1:
  117. Q->data[++k] = N.data[n++];
  118. break;
  119. }
  120. break;
  121. case 1:
  122. Q->data[++k] = N.data[n++];
  123. break;
  124. }
  125. }
  126. while(m<M.len){
  127. Q->data[++k] = M.data[m++];
  128. }
  129. while(n<N.len){
  130. Q->data[++k] = N.data[n++];
  131. }
  132. Q->len = k+ 1;
  133. if(k>MAXSIZE){
  134. printf( "ERROR!Out of bound!");
  135. return 0;
  136. }
  137. return 1;
  138. }
  139. /**矩阵减法**/
  140. //只需要将第二个矩阵的每个元素取反然后进行加法即可
  141. int SubMatrix(Matrix M,Matrix N,Matrix *Q){
  142. //相减成功返回1
  143. int len = N.len;
  144. for( int i= 0;i<len;i++){
  145. N.data[i].e = -N.data[i].e;
  146. }
  147. return AddMatrix(M,N,Q);
  148. }
  149. /**转置矩阵**/
  150. void TransposeMatrix(Matrix M,Matrix *N){
  151. int i ,k,t,col,*num,*position;
  152. num = ( int *) malloc((M.n+ 1)* sizeof( int));
  153. position = ( int *) malloc((M.n+ 1)* sizeof( int));
  154. N->m = M.m;
  155. N->n = M.n;
  156. N->len = M.len;
  157. if(N->len){
  158. for(col= 0;col<M.n;++col){
  159. num[col] = 0;
  160. }
  161. for(t = 0;t<M.len;t++){
  162. num[M.data[t].j]++;
  163. }
  164. position[ 0] = 0;
  165. for(col = 1;col<M.n;col++){
  166. position[col] = position[col -1]+num[col -1];
  167. }
  168. for(i = 0;i<M.len;i++){
  169. col = M.data[i].j;
  170. k = position[col];
  171. N->data[k].i =M.data[i].j;
  172. N->data[k].j = M.data[i].i;
  173. N->data[k].e = M.data[i].e;
  174. position[col]++;
  175. }
  176. }
  177. free(num);
  178. free(position);
  179. }
  180. void main(){
  181. Matrix m1,m2,m3,m4,m5;
  182. CreateMatrix(&m1);
  183. PrintMatrix(m1);
  184. CopyMatrix(m1,&m2);
  185. printf( "--------------\n\n");
  186. PrintMatrix(m2);
  187. AddMatrix(m1,m2,&m3);
  188. printf( "--------------\n\n");
  189. PrintMatrix(m3);
  190. SubMatrix(m3,m2,&m4);
  191. printf( "--------------\n\n");
  192. PrintMatrix(m4);
  193. TransposeMatrix(m4,&m5);
  194. printf( "--------------\n\n");
  195. PrintMatrix(m5);
  196. DestroyMatrix(&m1);
  197. printf( "--------------\n\n");
  198. PrintMatrix(m1);
  199. system( "pause");
  200. }





  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

深度学习推荐算法

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值