【数据结构】稀疏矩阵的快速转置

稀疏矩阵的快速转置

如图给出一个稀疏矩阵,要求表示出它的转置矩阵

由这个矩阵我们能轻松得到它的三元组顺序表

6行(x坐标) 7列(y坐标) 8个元素
1 2 12
1 3 9
3 1 -3
3 6 14
4 3 24
5 2 18
6 1 15
6 4 -7

接下来我们同样把转置后的矩阵的三元组顺序表表示出来

7行(x坐标) 6列(y坐标) 8个元素
1 3 -3
1 6 15
2 1 12
2 5 18
3 1 9
3 4 24
4 6 -7
6 3 14

我们要如何才能通过第一个三元组顺序表得到第二个三元组顺序表呢

按照最简单的思路,我们一般会通过双重循环

朴素的双重循环

按照列坐标从头到尾遍历,遇到1的时候,将i j互换放入到第二个三元组顺序表第一个位置

按照这样的方法,每一次都有可能需要遍历n次,一共需要走n趟,时间复杂度会是O(n²)级别


那么有没有一种方法能一步到位呢

快速转置

为实现快速转置,我们引入num数组和cpot数组

num[col]:表示矩阵M中第col列中非零元个数

cpot[col]:指示M中第col列第一个非零元在转置后的三元组顺序表中位置

显然有:

cpp 复制代码
cpot[1] = 1;
cpot[col] = cpot[col - 1] + num[col - 1]; //第col - 1列第一个非零元的位置加上第col - 1列非零元的个数

由此,我们能构建出col、num[col]、cpot[col]的一个表

col 1 2 3 4 5 6 7
num[col] 2 2 2 1 0 1 0
cpot[col] 1 3 5 7 8 8 9

这样我们遍历最开始的三元组顺序表

第一个列是2,我们查找 co l为 2 的 cpot[col] ,为 3,那么将原表第一行放入到转置后的第三个位置,同时 col 为 2 的 cpot[col] += 1

cpot[col] + 1 是因为若这一列还有非零元,那么肯定会是它的下一个位置

代码如下

cpp 复制代码
Status FastTransposeSMatrix( TSMatrix M, TSMatrix &T ) {  
    // 采用三元组顺序表存储表示,求稀疏矩阵 M 的转置矩阵 T
    T.mu = M.nu; T.nu = M.mu; T.tu = M.tu; 
    if (T.tu) { 
        // mu行 nu列
        for (col=1; col<=M.nu; ++col)    num[col] = 0; 
        for (t=1; t<=M.tu; ++t)   ++ num[M.data[t].j];  
                 // 求 M 中各列非零元的个数
        cpot[1] = 1;
        for (col=2; col<=M.nu; ++col)  cpot[col] = cpot[col -1] + num[col -1]; 
        // 求 M 中各列的第一个非零元在 T.data 中的序号 
            for (p=1; p<=M.tu; ++p) {  // 转置矩阵元素  
                col = M.data[p].j;    
                q = cpot[col]; 
                T.data[q].i = M.data[p].j;    
                T.data[q].j = M.data[p].i; 
               	T.data[q].e = M.data[p].e;   
                ++ cpot[col];  
            } // for
    } // if
    return OK;
} // FastTransposeSMatrix
相关推荐
做怪小疯子2 小时前
LeetCode 热题 100——矩阵——旋转图像
算法·leetcode·矩阵
sin_hielo3 小时前
leetcode 2435
数据结构·算法·leetcode
crescent_悦3 小时前
PTA L1-020 帅到没朋友 C++
数据结构·c++·算法
稚辉君.MCA_P8_Java5 小时前
Gemini永久会员 Java动态规划
java·数据结构·leetcode·排序算法·动态规划
cookqq6 小时前
mongodb根据索引IXSCAN 查询记录流程
数据结构·数据库·sql·mongodb·nosql
ohyeah7 小时前
栈:那个“先进后出”的小可爱,其实超好用!
前端·数据结构
历程里程碑8 小时前
各种排序法大全
c语言·数据结构·笔记·算法·排序算法
passxgx9 小时前
11.1 高斯消元法的应用
线性代数·矩阵
embrace999 小时前
【C语言学习】结构体详解
android·c语言·开发语言·数据结构·学习·算法·青少年编程
稚辉君.MCA_P8_Java10 小时前
通义 Go 语言实现的插入排序(Insertion Sort)
数据结构·后端·算法·架构·golang