leetcode 1292(二维前缀和)

1292: 元素和小于等于阈值的正方形的最大边长

思路1:一维前缀和+暴力枚举

复制代码
class Solution {
public:
    int maxSideLength(vector<vector<int>>& mat, int threshold) {
        int m=mat.size(),n=mat[0].size();
        vector<vector<int>> rowPrefix(m+1,vector<int>(n+1));
        for(int i=0;i<m;i++){
            for(int j=0;j<n;j++){
                rowPrefix[i+1][j+1]=rowPrefix[i+1][j]+mat[i][j];
            }
        }
        for(int k=min(m,n);k>0;k--){
            for(int r=0;r+k<=m;r++){
                for(int c=0;c+k<=n;c++){ //枚举正方形左上角(r,c)
                    long long sum=0;
                    for(int i=r;i<r+k;i++){
                        sum+=rowPrefix[i+1][c+k]-rowPrefix[i+1][c];
                        if(sum>threshold) break;
                    }
                    if(sum<=threshold){
                        return k;
                    }
                }
            }
        }
        return 0;
    }
};

思路2:二维前缀和

预处理二维前缀和后,暴力的做法是写一个三重循环:

  • 外面两重循环,枚举正方形的左上角 (i,j)。
  • 最内层循环,枚举正方形的边长为 1,2,3,... 直到出界或者正方形元素和超过 threshold 为止。在此过程中更新答案 ans 的最大值。

这样做的时间复杂度是 O(mn min(m,n))。

只需改一个地方,就能让算法的时间复杂度变小:

  • 最内层循环,从 ans+1 开始枚举。

比如现在 ans=3,那么枚举正方形的边长为 1,2,3 是毫无意义的,不会让答案变得更大。所以直接从 ans+1=4 开始枚举更好。

复制代码
class Solution {
public:
    int maxSideLength(vector<vector<int>>& mat, int threshold) {
        int m=mat.size(),n=mat[0].size();
        vector sum(m+1,vector<int>(n+1));
        for(int i=0;i<m;i++){
            for(int j=0;j<n;j++){
                sum[i+1][j+1]=sum[i+1][j]+sum[i][j+1]-sum[i][j]+mat[i][j];
            }
        }
        //返回左上角在(r1, c1),右下角在(r2, c2)的子矩阵元素和
        auto query=[&](int r1,int c1,int r2,int c2)->int{
            return sum[r2+1][c2+1]-sum[r2+1][c1]-sum[r1][c2+1]+sum[r1][c1];
        };

        int ans=0;
        for(int i=0;i<m;i++){
            for(int j=0;j<n;j++){
                //枚举边长为ans+1的正方形,左上角在(i, j),右下角在(i+ans, j+ans)
                while(i+ans<m && j+ans<n && query(i,j,i+ans,j+ans)<=threshold){
                    ans++; //确认ans+1可行后做ans++,即返回正确答案
                }
            }
        }

        return ans;
    }
};
相关推荐
灵感__idea3 小时前
Hello 算法:众里寻她千“百度”
前端·javascript·算法
Wect13 小时前
LeetCode 130. 被围绕的区域:两种解法详解(BFS/DFS)
前端·算法·typescript
NAGNIP1 天前
一文搞懂深度学习中的通用逼近定理!
人工智能·算法·面试
颜酱1 天前
单调栈:从模板到实战
javascript·后端·算法
CoovallyAIHub2 天前
仿生学突破:SILD模型如何让无人机在电力线迷宫中发现“隐形威胁”
深度学习·算法·计算机视觉
CoovallyAIHub2 天前
从春晚机器人到零样本革命:YOLO26-Pose姿态估计实战指南
深度学习·算法·计算机视觉
CoovallyAIHub2 天前
Le-DETR:省80%预训练数据,这个实时检测Transformer刷新SOTA|Georgia Tech & 北交大
深度学习·算法·计算机视觉
CoovallyAIHub2 天前
强化学习凭什么比监督学习更聪明?RL的“聪明”并非来自算法,而是因为它学会了“挑食”
深度学习·算法·计算机视觉
CoovallyAIHub2 天前
YOLO-IOD深度解析:打破实时增量目标检测的三重知识冲突
深度学习·算法·计算机视觉
NAGNIP2 天前
轻松搞懂全连接神经网络结构!
人工智能·算法·面试