【2024.3.19练习】统计子矩阵

题目描述


题目分析

这道题一开始没有思路,使用蛮力枚举的方法时间复杂度为O(N^3M^3),显然超时。

参考题解后学会了化二维问题为一维问题,先使用O(N^2)的复杂度限制子矩阵的高度,再考虑列,这样就将子矩阵的和问题转变为了连续子序列的和问题,显然可以用双指针法减低复杂度。因此总时间复杂度减低为了O(N^2M),看似非常大,但是由于循环体内语句已经十分简短,运行时间可以控制在百毫秒级,不会导致超时。

注意,需要提前使用动态规划的思路算出每列的数字和,不要在循环体内临时计算,否则仍会运行超时。


我的代码

这道题的坑在于虽然K限制在int范围内,但ans的值最大为500^4/2,会超出int范围!因此使用long long存储数据。

#include <iostream>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll; 
int n;
int m;
int k;
const int max_n = 502;
int A[max_n][max_n];
int sum[max_n][max_n];
int main() {
	//初始化 
	int i = 0;
	int j = 0;
	cin >> n >> m >> k;
	for(i = 0;i <= n + 1;i++){
		for(j = 0;j <= m + 1;j++){
			A[i][j] = 0;
			sum[i][j] = 0;
		}
	}
	for(i = 1;i <= n;i++){
		for(j = 1;j <= m;j++){
			cin>>A[i][j];
			sum[i][j] = sum[i-1][j]+A[i][j];
		}
	}
	//降维操作
	ll ans = 0;
	for(i = 1;i <= n;i++){
		for(j = 1;j <= i;j++){
			//尺取法
			int s = 1;
			int t = 1;
			int flag = 0;
			int sum2 = sum[i][1] - sum[j-1][1];
			for( ;flag == 0; ){
				if(sum2 <= k){
					ans = ans + t - s + 1;
					//cout<<s<<"-"<<t<<":"<<sum2<<endl;
					t++;
					sum2 = sum2 + (sum[i][t] - sum[j-1][t]);
				}
				else{
					if(s == t){
						//cout<<s<<"-"<<t<<":"<<sum2<<endl;
						t++;
						sum2 = sum2 + (sum[i][t] - sum[j-1][t]);
					}else{
						//cout<<s<<"-"<<t<<":"<<sum2<<endl;
						sum2 = sum2 - (sum[i][s] - sum[j-1][s]);
						s++;
					}
				}
				if(t == m + 1){
					flag = 1;
				}	
			}
			//cout<<"ans:"<<ans<<endl;
		}
	}
	//获取答案
	cout<<ans; 
	return 0;
}
  • 8
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值