LeetCode 363:Max Sum of Rectangle No Larger Than K
题目链接
链接:https://leetcode.com/problems/max-sum-of-rectangle-no-larger-than-k/description/
题解&代码
1、暴力枚举所有的情况,时间复杂度O(n2*m2),实际耗时759 ms
class Solution {
public:
int maxSumSubmatrix(vector<vector<int> >& matrix, int k) {
int n=matrix.size(),m=matrix[0].size();
vector<vector<int> > sumv(n+1,vector<int>(m+1,0));
for(int i=1; i<=n; i++) {
for(int j=1; j<=m; j++) {
sumv[i][j]=matrix[i-1][j-1]+sumv[i-1][j]+sumv[i][j-1]-sumv[i-1][j-1];
}
}
int ans=-INF;
for(int i1=1;i1<=n;i1++){
for(int j1=1;j1<=m;j1++){
for(int i2=i1;i2<=n;i2++){
for(int j2=j1;j2<=m;j2++){
int val=sumv[i2][j2]-sumv[i1-1][j2]-sumv[i2][j1-1]+sumv[i1-1][j1-1];
if(val<=k){
ans=max(ans,val);
}
}
}
}
}
return ans;
}
private:
static const int INF=0x3f3f3f3f;
};
2、先通过枚举一维进行降维,然后再暴力枚举所有区间,时间复杂度O(m2*n2),实际耗时736ms。
class Solution {
public:
int maxSumSubmatrix(vector<vector<int> >& matrix, int k) {
int n=matrix.size(),m=matrix[0].size();
vector<vector<int> > sumv(n+1,vector<int>(m+1,0));
for(int i=1; i<=n; i++) {
for(int j=1; j<=m; j++) {
sumv[i][j]=sumv[i][j-1]+matrix[i-1][j-1];
}
}
int ans=-INF;
vector<int> arr(n+1);
for(int len=1; len<=m; len++) {
for(int j=len; j<=m; j++) {
arr[0]=0;
for(int i=1; i<=n; i++) {
int val=sumv[i][j]-sumv[i][j-len];
arr[i]=val+arr[i-1];
}
for(int i=1;i<=n;i++){
for(int j=0;j<i;j++){
int x=arr[i]-arr[j];
if(x<=k) ans=max(ans,x);
}
}
}
}
return ans;
}
private:
static const int INF=0x3f3f3f3f;
};
3、 第2种方法基础上加一个剪枝,在枚举区间之前,先用dp(O(n))求出最大值区间和最小值区间,然后判断k是否在区间内,时间复杂度:‘玄学’,实际耗时19ms。
class Solution {
public:
int maxSumSubmatrix(vector<vector<int> >& matrix, int k) {
int n=matrix.size(),m=matrix[0].size();
vector<vector<int> > sumv(n+1,vector<int>(m+1,0));
for(int i=1; i<=n; i++) {
for(int j=1; j<=m; j++) {
sumv[i][j]=sumv[i][j-1]+matrix[i-1][j-1];
}
}
int ans=-inf;
vector<int> arr(n+1);
for(int len=1; len<=m; len++) {
for(int j=len; j<=m; j++) {
arr[0]=0;
LL mi=inf,ma=-inf;
LL pre=inf,aft=-inf;
for(int i=1; i<=n; i++) {
int val=sumv[i][j]-sumv[i][j-len];
arr[i]=val+arr[i-1];
pre=min(pre+val,(LL)val);
aft=max(aft+val,(LL)val);
mi=min(mi,pre);
ma=max(ma,aft);
}
if(k<mi) continue;
else if(k==mi) {
ans=k;
} else if(k>=ma) {
ans=max((LL)ans,ma);
} else {
for(int i=1; i<=n; i++) {
for(int j=0; j<i; j++) {
int x=arr[i]-arr[j];
if(x<=k) ans=max(ans,x);
}
}
if(ans==k) return ans;
}
}
}
return ans;
}
private:
static const int inf=0x7fffffff;
typedef long long LL;
};
4、正解:先降维,然后枚举区间终点,用set维护前缀和,用lower_bound去查区间起点,从而O(nlogn)的复杂度解决该问题的一维版本。加上降维的时间复杂度为O(m^2*nlogn)。实际耗时:259ms。
class Solution {
public:
int maxSumSubmatrix(vector<vector<int> >& matrix, int k) {
int n=matrix.size(),m=matrix[0].size();
vector<vector<int> > sumv(n+1,vector<int>(m+1,0));
for(int i=1; i<=n; i++) {
for(int j=1; j<=m; j++) {
sumv[i][j]=sumv[i][j-1]+matrix[i-1][j-1];
}
}
int ans=-INF;
vector<int> pre(n+1),arr(n+1);
set<pair<int,int> > myset;
set<pair<int,int> >::iterator it;
for(int len=1; len<=m; len++) {
for(int j=len; j<=m; j++) {
pre[0]=0;
myset.clear();
for(int i=1; i<=n; i++) {
arr[i]=sumv[i][j]-sumv[i][j-len];
pre[i]=arr[i]+pre[i-1];
myset.insert(make_pair(pre[i],i));
}
for(int i=1;i<=n;i++){
if(arr[i]<=k) ans=max(ans,arr[i]);
int key=k-arr[i]+pre[i];
it=myset.lower_bound(make_pair(key,0x7fffffff));
if(it!=myset.begin()){
it--;
int tmp=it->first;
ans=max(ans,tmp-pre[i]+arr[i]);
}
myset.erase(make_pair(pre[i],i));
}
}
}
return ans;
}
private:
static const int INF=0x7fffffff;
};
5、正解优化版本(空间优化+常数优化)实际耗时:162ms。
class Solution {
public:
int maxSumSubmatrix(vector<vector<int> >& matrix, int k) {
int n=matrix.size(),m=matrix[0].size();
int mi=min(n,m),ma=max(n,m);
bool isColMax=m>n;
int ans=-2147483648;
for(int st=1; st<=mi; st++) {
vector<int> arr(ma+1,0);
for(int ed=st; ed<=mi; ed++) {
set<int> myset;
int cnt=0;
myset.insert(cnt);
for(int i=1; i<=ma; i++) {
arr[i]+=isColMax?matrix[ed-1][i-1]:matrix[i-1][ed-1];
cnt+=arr[i];
set<int>::iterator it=myset.lower_bound(cnt-k);
if(it!=myset.end()) ans=max(ans,cnt-*it);
myset.insert(cnt);
}
}
}
return ans;
}
};
6、方法5+方法3提到的dp剪枝:实际耗时:19ms。
class Solution {
public:
int maxSumSubmatrix(vector<vector<int> >& matrix, int k) {
int n=matrix.size(),m=matrix[0].size();
int mi=min(n,m),ma=max(n,m);
bool isColMax=m>n;
int ans=-2147483648;
for(int st=1; st<=mi; st++) {
vector<int> arr(ma+1,0);
for(int ed=st; ed<=mi; ed++) {
set<int> myset;
int cnt=0;
myset.insert(cnt);
for(int i=1; i<=ma; i++) {
arr[i]+=isColMax?matrix[ed-1][i-1]:matrix[i-1][ed-1];
cnt+=arr[i];
set<int>::iterator it=myset.lower_bound(cnt-k);
if(it!=myset.end()) ans=max(ans,cnt-*it);
myset.insert(cnt);
}
}
}
return ans;
}
};
LeetCode 363:Max Sum of Rectangle No Larger Than K的更多相关文章
- 363. Max Sum of Rectangle No Larger Than K
/* * 363. Max Sum of Rectangle No Larger Than K * 2016-7-15 by Mingyang */ public int maxSumSubmatri ...
- [LeetCode] 363. Max Sum of Rectangle No Larger Than K 最大矩阵和不超过K
Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...
- 【LeetCode】363. Max Sum of Rectangle No Larger Than K 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题目地址: https://leetcode.com/problems/max-sum- ...
- 【leetcode】363. Max Sum of Rectangle No Larger Than K
题目描述: Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the ma ...
- [LeetCode] Max Sum of Rectangle No Larger Than K 最大矩阵和不超过K
Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...
- Leetcode: Max Sum of Rectangle No Larger Than K
Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...
- 363 Max Sum of Rectangle No Larger Than K 最大矩阵和不超过K
Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...
- [Swift]LeetCode363. 矩形区域不超过 K 的最大数值和 | Max Sum of Rectangle No Larger Than K
Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...
- Max Sum of Rectangle No Larger Than K
Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...
随机推荐
- jQuery EasyUI combobox多选及赋值
1.多选属性multiple multiple:true多选 multiple:false单选 <input class="easyui-combobox" id=&qu ...
- C#异步编程の----Threadpool( 线程池)
简介: 一个托管线程的创建需要数千个CPU周期,并且当发生线程切换时也会带来明显的开销.考虑线程的重用,避免不断重复创建新的线程是提高系统效率的一种方式. 线程池是一种提供效率的方式,它创建好一些线程 ...
- matlab数字图像处理 入门基础
本代码基于张铮主编的<数字图像处理与机器视觉>一书. 源图片:lena A = imread ('lena.jpg'); %读入图像lena.jpg,赋给变量A %imwrite(A,'l ...
- CSS3系列教程:HSL 和HSL
使用CSS3 HSL声明同样是用来设置颜色的.下一个呢? HSLA? 是的,这个和RGBA的效果是一样的. HSL声明使用色调Hue(H).饱和度Saturation(s)和亮度Lightness(L ...
- 简单使用metamascara
package.json 配置文件信息 { "name": "test-mascara", "version": "1.0.0&q ...
- 基于Matlab的多自由度系统固有频率及振型计算
可参考文涛,基于Matlab语言的多自由度振动系统的固有频率及主振型计算分析,2007 对于无阻尼系统 [VEC,VAL]=eig(inv(A)*K) 对于有阻尼系统,参考振动论坛计算程序 输入M,D ...
- Daily Sentence(英语每日一句)
1.When you want to give up, remember why you started. 当你想要放弃的时候,请记住当初你为何而开始. 2.It does not do to dwe ...
- Android学习之触点事件的处理
知识点: 1. Android开发中的运动事件:触摸屏(TouchScreen)和滚动球(TrackBall) 2.对运动事件的处理:MotionEvent 3.触摸时必发的三个MotionEvent ...
- java 面向对象抽象类和接口
1.abstract:可以修饰类和方法,被abstract修饰过的称为抽象类和抽象方法. 抽象类: 除了不可以创建对象, 其他和普通类一致. 可以有成员方法.静态方法和构造方法(提供给子类调用)等. ...
- LVDS原理及设计指南--以及衍生的B-LVDS-M-LVDS--CML-LVPECL电平等
LVDS是一种低摆幅的差分信号技术,它使得信号能在差分PCB 线对或平衡电缆上以几百Mbps的速率传输,其低压幅和低电流驱动输出实现了低噪声和低功耗. IEEE 在两个标准中对LVDS 信号 ...