C++前缀和算法:构造乘积矩阵

基础知识点

C++算法:前缀和基础

题目

给你一个下标从 0 开始、大小为 n * m 的二维整数矩阵 grid ,定义一个下标从 0 开始、大小为 n * m 的的二维矩阵 p。如果满足以下条件,则称 p 为 grid 的 乘积矩阵 :

对于每个元素 pij ,它的值等于除了 gridij 外所有元素的乘积。乘积对 12345 取余数。

返回 grid 的乘积矩阵。

示例 1:

输入:grid = \[1,2,3,4]

输出:\[24,12,8,6]

解释:p00 = grid01 * grid10 * grid11 = 2 * 3 * 4 = 24

p01 = grid00 * grid10 * grid11 = 1 * 3 * 4 = 12

p10 = grid00 * grid01 * grid11 = 1 * 2 * 4 = 8

p11 = grid00 * grid01 * grid10 = 1 * 2 * 3 = 6

所以答案是 \[24,12,8,6] 。

示例 2:

输入:grid = \[12345,2,1]

输出:\[2,0,0]

解释:p00 = grid01 * grid02 = 2 * 1 = 2

p01 = grid00 * grid02 = 12345 * 1 = 12345. 12345 % 12345 = 0 ,所以 p01 = 0

p02 = grid00 * grid01 = 12345 * 2 = 24690. 24690 % 12345 = 0 ,所以 p02 = 0

所以答案是 \[2,0,0] 。

感悟

原以为和MOD = 1000000007一样,直接使用封装好的此类,发现错误。赛场上时间紧急,来不及分析是两个不同的问题,还是我封装错误。只好使用笨办法。我记得1000000007是质数,才能转除为乘。考虑过12345是否是质数,当时觉判断烦恼,所以没判断。现在觉得很简单:以5结尾,就是5的倍数,不是质数。

分析

前缀和后缀和。第一轮的preRow记录0,r) 所有元素的乘积,vLeft\[rc 记录本行左边各元素的乘积,vRightrc记录本行右边各元素的乘积。vRetrc记录这三个的乘积。第二轮preRow记录r+1,m_r)所有元素的乘积。第二轮vRet\[rc乘以preRow就是结果。

测试用例

1 2 3
4 5 6
7 8 9

结果

当前数 前面行的乘积 前面行的乘积 左边乘积 右边乘积
1 1 4...9 1 6
2 1 4...9 1 3
3 1 4...9 2 1
4 6 7...9 1 30
5 6 7...9 4 6
6 6 7...9 20 1
7 1...6 1 1 72
8 1...6 1 7 9
9 1...6 1 56 1

解释

对{4,5,6}而言第一轮preRow是12 3=6,第二轮preRow是789。对4而言,left是1,right是30。对5而言,left是4,right是6。对6而言,left是20,right是1。

时间复杂度

O(n^2) 2轮,每轮2层循环,每层循环是O(n)。

代码

class Solution {

public:

vector<vector> constructProductMatrix(vector<vector>& grid) {

m_r = grid.size();

m_c = grid.front().size();

//vLeft记录当前行,左边的成绩

vector<vector> vLeft(m_r, vector(m_c)), vRight(m_r, vector(m_c)), vRet(m_r, vector(m_c));

int iPreRow = 1;

for (int r = 0; r < m_r; r++)

{

int pre = 1;

for (int c = 0; c < m_c; c++)

{

vLeftrc = pre;

MulSelf(pre, gridrc);

}

pre = 1;

for (int c = m_c-1 ; c >= 0 ; c-- )

{

vRightrc = pre;

MulSelf(pre, gridrc);

}

for (int c = 0; c < m_c; c++)

{

vRetrc = 1;

MulSelf(vRetrc, iPreRow);

MulSelf(vRetrc, vLeftrc);

MulSelf(vRetrc, vRightrc);

}

MulSelf(iPreRow, pre);

}

iPreRow = 1;

for (int r = m_r-1; r >= 0 ; r-- )

{

int pre = 1;

for (int c = 0; c < m_c; c++)

{

MulSelf(vRetrc, iPreRow);

MulSelf(pre, gridrc);

}

MulSelf(iPreRow, pre);

}

return vRet;

}

void MulSelf(int& self, int other)

{

const int MOD = 12345;

self = ((long long)self * other) % MOD;

}

int m_r, m_c;

};

一维化降低复杂度

分析

vLeftrc记录 [0,r)行所有元素及r行[0,c)列元素的乘积,第二轮的pre记录(r,m_c)行所有元素及r行(c,m_c)列元素的乘积。

vLeft11 = 12 34 第二轮的pre = 9 876

代码

class Solution {

public:

vector<vector> constructProductMatrix(vector<vector>& grid) {

m_r = grid.size();

m_c = grid.front().size();

vector < vector> vLeft(m_r, vector(m_c));

int pre = 1;

for (int r = 0; r < m_r; r++)

{

for (int c = 0; c < m_c; c++)

{

vLeftrc = pre;

MulSelf(pre, gridrc);

}

}

vector<vector> vRet(m_r, vector(m_c));

pre = 1;

for (int r = m_r-1 ; r >= 0 ;r--)

{

for (int c = m_c-1 ; c >= 0 ; c-- )

{

const int index = m_c * r + c;

vRetrc = pre;

MulSelf(vRetrc, vLeftrc);

MulSelf(pre, gridrc);

}

}

return vRet;

}

void MulSelf(int& self, int other)

{

const int MOD = 12345;

self = ((long long)self * other) % MOD;

}

int m_r, m_c;

};

测试用例

template

void Assert(const vector& v1, const vector& v2)

{

if (v1.size() != v2.size())

{

assert(false);

return;

}

for (int i = 0; i < v1.size(); i++)

{

assert(v1i == v2i);

}

}

template

void Assert(const T& t1, const T& t2)

{

assert(t1 == t2);

}

int main()

{

vector<vector>grid = { {1,2,3},{4,5,6} };

vector<vector> ans = { {720,360,240},{180,144,120} };

auto res = Solution().constructProductMatrix(grid);

Assert(res, ans);

grid = { {1,2,},{3,4},{5,6 }};

ans = { {720,360},{240,180},{144,120} };

res = Solution().constructProductMatrix(grid);

Assert(res, ans);

grid = { { 1,2,3,4,5,6 } };

ans = { { 720,360,240,180,144,120} };

res = Solution().constructProductMatrix(grid);

Assert(res, ans);

grid = { { 1},{2},{3},{4},{5},{6} };

ans = { { 720},{360},{240},{180},{144},{120} };

res = Solution().constructProductMatrix(grid);

Assert(res, ans);

CConsole::Out(res);

}

其它

视频课程

要是你认为本篇难道较大,不好入手,推荐你先学习基础算法的课程,我已完成部分,余下部分持续更新中,就在CSDN学院。
https://edu.csdn.net/course/detail/38771

C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

测试环境

操作系统:win7 开发环境: VS2019 C++17

相关下载

如果你想观其大略,建设下载《闻缺陷则喜算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

博主想队大家说的话
墨家名称的来源:有所得以墨记之。
闻缺陷则喜的来由:早发现,早修改问题,成本更低
程序是龙,算法是睛
相关推荐
elseif1234 小时前
【C++】vector 详细版
开发语言·c++·算法
变量未定义~4 小时前
既约分数、阶乘约数、逆元、最大质因子个数【算法赛】
算法
cany10004 小时前
C++ -- 原子变量
c++
KaMeidebaby4 小时前
卡梅德生物技术快报|Western Blot 实验应用:肺肠轴机制研究全流程技术解析
前端·数据库·人工智能·算法·百度
2601_957888565 小时前
从数据隔离到全链路分发:短视频矩阵系统的防关联底层逻辑与提效实践
线性代数·矩阵
AhriProGramming5 小时前
计算机科普故事会-<2>见微知著
算法
cany10005 小时前
C++ -- 队列std::queue
开发语言·c++
周末也要写八哥5 小时前
C++中单线程方式之无脑上锁
java·开发语言·c++
cany10005 小时前
C++ -- 动态内存分配和释放(new/delete)
开发语言·c++