C++前缀和算法:构造乘积矩阵
最佳答案 问答题库738位专家为你答疑解惑
题目
给你一个下标从 0 开始、大小为 n * m 的二维整数矩阵 grid ,定义一个下标从 0 开始、大小为 n * m 的的二维矩阵 p。如果满足以下条件,则称 p 为 grid 的 乘积矩阵 :
对于每个元素 p[i][j] ,它的值等于除了 grid[i][j] 外所有元素的乘积。乘积对 12345 取余数。
返回 grid 的乘积矩阵。
示例 1:
输入:grid = [[1,2],[3,4]]
输出:[[24,12],[8,6]]
解释:p[0][0] = grid[0][1] * grid[1][0] * grid[1][1] = 2 * 3 * 4 = 24
p[0][1] = grid[0][0] * grid[1][0] * grid[1][1] = 1 * 3 * 4 = 12
p[1][0] = grid[0][0] * grid[0][1] * grid[1][1] = 1 * 2 * 4 = 8
p[1][1] = grid[0][0] * grid[0][1] * grid[1][0] = 1 * 2 * 3 = 6
所以答案是 [[24,12],[8,6]] 。
示例 2:
输入:grid = [[12345],[2],[1]]
输出:[[2],[0],[0]]
解释:p[0][0] = grid[0][1] * grid[0][2] = 2 * 1 = 2
p[0][1] = grid[0][0] * grid[0][2] = 12345 * 1 = 12345. 12345 % 12345 = 0 ,所以 p[0][1] = 0
p[0][2] = grid[0][0] * grid[0][1] = 12345 * 2 = 24690. 24690 % 12345 = 0 ,所以 p[0][2] = 0
所以答案是 [[2],[0],[0]] 。
感悟
原以为和MOD = 1000000007一样,直接使用封装好的此类,发现错误。赛场上时间紧急,来不及分析是两个不同的问题,还是我封装错误。只好使用笨办法。
分析
前缀和后缀和。第一轮的preRow记录[0,r) 所有元素的乘积,vLeft[r][c] 记录本行左边各元素的乘积,vRight[r][c]记录本行右边各元素的乘积。vRet[r][c]记录这三个的乘积。第二轮preRow记录[r+1,m_r)所有元素的乘积。第二轮vRet[r][c]乘以preRow就是结果。
时间复杂度
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<int>> vLeft(m_r, vector<int>(m_c)), vRight(m_r, vector<int>(m_c)), vRet(m_r, vector<int>(m_c));int iPreRow = 1; for (int r = 0; r < m_r; r++){int pre = 1;for (int c = 0; c < m_c; c++){vLeft[r][c] = pre;MulSelf(pre, grid[r][c]);}pre = 1;for (int c = m_c-1 ; c >= 0 ; c-- ){vRight[r][c] = pre;MulSelf(pre, grid[r][c]);}for (int c = 0; c < m_c; c++){vRet[r][c] = 1;MulSelf(vRet[r][c], iPreRow);MulSelf(vRet[r][c], vLeft[r][c]);MulSelf(vRet[r][c], vRight[r][c]);}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(vRet[r][c], iPreRow); MulSelf(pre, grid[r][c]);} 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;
};
其它
视频课程
要是你认为本篇难道较大,不好入手,推荐你先学习基础算法的课程,我已完成部分,余下部分持续更新中,就在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
99%的人还看了
相似问题
- C语言矩阵乘积(ZZULIOJ1127:矩阵乘积)
- 卡尔曼家族从零解剖-(07) 高斯分布积分为1,高斯分布线性变换依旧为高斯分布,两高斯函数乘积仍为高斯。
- 行列式某一行所有元素除(i, j)元外都为0,则行列式等于(i, j)元与(i, j)元的代数余子式的乘积
- 矩阵乘积的迹对矩阵求导
- LeetCode:318. 最大单词长度乘积(C++)
- 【每日一题Day370】LC318最大单词长度乘积 | 哈希表 位运算
- LeetCode 318. 最大单词长度乘积
- 【教3妹学编程-算法题】最大单词长度乘积
- leetcode 238. 除自身以外数组的乘积
- 华为OD 无重复字符的元素长度乘积的最大值(100分)【java】B卷
猜你感兴趣
版权申明
本文"C++前缀和算法:构造乘积矩阵":http://eshow365.cn/6-19626-0.html 内容来自互联网,请自行判断内容的正确性。如有侵权请联系我们,立即删除!