leetcode 2536

2536: 子矩阵元素加1

思路1:暴力解

复制代码
class Solution {
public:
    vector<vector<int>> rangeAddQueries(int n, vector<vector<int>>& queries) {
        vector<vector<int>> ans(n,vector<int>(n,0));
        for(auto& qur :queries){
            for(int i=qur[0];i<=qur[2];i++){
                for(int j=qur[1];j<=qur[3];j++){
                    ans[i][j]++;
                }
            }
        }

        return ans;
    }
};

思路2:二维差分+二维前缀和

一维差分,见 leetcode 2528

使用二维差分数组diff,其中diff[i][j]表示原数组mat[i][j]到mat[n-1][n-1]每个元素++变化值。++

例如将mat[x1][y1]到mat[x2][y2]每个元素值加一,反映到差分数组上就表现为:

diff[x1][y1]+=1,先从mat[x1][y1]到mat[n-1][n-1]每个元素都加一

diff[x2+1][y1]−=1,由于从mat[x2+1][y1]到mat[n-1][n-1]每个元素无需改变,第一步加的范围太大了,不该变的撤销变化

diff[x1][y2+1]−=1,与2同理,由于从mat[x1][y2+1]到mat[n-1][n-1]每个元素无需改变,第一步加的范围太大了,不该变的撤销变化

diff[x2+1][y2+1]+=1,由于第二步和第三步都包含范围mat[x2+1][y2+1],所以这块"重复撤销"了,再加回来

对于每个query,以O(1)的时间复杂度可以计算出diff数组的变化。最终处理完所有query,再由diff差分数组反推出原数组mat就好了。

代码实现时,为方便计算二维前缀和,可以在二维差分矩阵最上面添加一行 0,最左边添加一列 0,这样计算二维前缀和无需考虑下标越界。

  • 公式
    diff[i][j] = diff[i][j] + diff[i-1][j] + diff[i][j-1] - diff[i-1][j-1]

    正是二维前缀和递推式。

    class Solution {
    public:
    vector<vector<int>> rangeAddQueries(int n, vector<vector<int>>& queries) {
    /*二维差分,记录差分数组的变化
    二维矩阵a的区域+1操作,等价于更新差分矩阵d中4个位置的差分值。
    为方便计算二维前缀和,在二维差分矩阵上下左右各添加一行0 */
    vector<vector<int>> diff(n+2,vector<int>(n+2)); //防止越界
    for(auto& q :queries){
    int r1=q[0],c1=q[1],r2=q[2],c2=q[3];
    diff[r1+1][c1+1]++;
    diff[r1+1][c2+2]--;
    diff[r2+2][c1+1]--;
    diff[r2+2][c2+2]++;
    }

    复制代码
          // 原地计算 diff 的二维前缀和,然后填入答案
          vector<vector<int>> ans(n,vector<int>(n));
          for(int i=0;i<n;i++){
              for(int j=0;j<n;j++){  
                  diff[i+1][j+1]+=diff[i][j+1]+diff[i+1][j]-diff[i][j];
                  ans[i][j]=diff[i+1][j+1];
              }
          }
    
          return ans;
      }

    };

相关推荐
power 雀儿1 小时前
掩码(Mask)机制 结合 多头自注意力函数
算法
会叫的恐龙1 小时前
C++ 核心知识点汇总(第六日)(字符串)
c++·算法·字符串
小糯米6011 小时前
C++顺序表和vector
开发语言·c++·算法
We་ct1 小时前
LeetCode 56. 合并区间:区间重叠问题的核心解法与代码解析
前端·算法·leetcode·typescript
Lionel6892 小时前
分步实现 Flutter 鸿蒙轮播图核心功能(搜索框 + 指示灯)
算法·图搜索算法
小妖6662 小时前
js 实现快速排序算法
数据结构·算法·排序算法
xsyaaaan2 小时前
代码随想录Day30动态规划:背包问题二维_背包问题一维_416分割等和子集
算法·动态规划
zheyutao3 小时前
字符串哈希
算法
A尘埃3 小时前
保险公司车险理赔欺诈检测(随机森林)
算法·随机森林·机器学习
大江东去浪淘尽千古风流人物4 小时前
【VLN】VLN(Vision-and-Language Navigation视觉语言导航)算法本质,范式难点及解决方向(1)
人工智能·python·算法