二维前缀和模板

二维前缀和模板

题目描述:

输入一个 n 行 m 列的整数矩阵,再输入 q个询问,每个询问包含四个整数 x1,y1,x2,y2,表示一个子矩阵的左上角坐标和右下角坐标。

对于每个询问输出子矩阵中所有数的和。

输入格式:

第一行包含三个整数 n,m,q

接下来 n 行,每行包含 m 个整数,表示整数矩阵。

接下来 q行,每行包含四个整数 x1,y1,x2,y2,表示一组询问。

输出格式:

共 q行,每行输出一个询问的结果。

数据范围:

1≤n,m≤1000
1≤q≤200000
1≤x1≤x2≤n
1≤y1≤y2≤m
−1000≤矩阵内元素的值≤1000

输入样例:

3 4 3
1 7 2 4
3 6 2 8
2 1 2 3
1 1 2 2
2 1 3 4
1 3 3 4

输出样例:

17
27
21

解题思路:

我们首先要理解,sum[i,j]是什么意思。

image

如图所示,sum[i,j]就是红色区域所包含的所有数之和.

但是题目要求我们把,[x1,y1]到[x2,y2]区域内数之和求出来,所以我们可以再画一个图.

image

如图所示,我们要求[x1,y1]到[x2,y2]内所有数之和,是不是可以让sum[x2,y2]减去黄色区域之和,也就是sum[x2,y2]-sum[x1-1,y2]-sum[x2,y1-1],但是我们会发现,这样会使蓝色区域被减两边,所以我们还需要把蓝色区域加一遍,最终表达式也就是

sum[x2,y2]-sum[x1-1,y2]-sum[x2,y1-1]+sum[x1-1,y1-1]

还有一步,我们要处理这个前缀和。

image

如图,我们要求sum[i,j],先让黄色区域加起来,也就是sum[i-1,j]+sum[i,j-1],这样会让蓝色区域加重,再减去蓝色区域,也就是sum[i-1,j-1],最后再加上a[i,j]就是我们要求的式子。

表达式如下:

sum[i][j]=sum[i-1][j]+sum[i][j-1]-sum[i-1][j-1]+a[i][j]

代码如下:

#include <bits/stdc++.h>
using namespace std;
const int N=1e3+10;
int n,m,q;
int a[N][N],sum[N][N];
int main()
{
    scanf("%d%d%d",&n,&m,&q);
	
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			scanf("%d",&a[i][j]);
		}
	}
	
	//处理前缀和 
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			 sum[i][j]=sum[i-1][j]+sum[i][j-1]-sum[i-1][j-1]+a[i][j];
		}
	}
	
	//询问 
	while(q--)
	{
		int x1,y1,x2,y2;
		scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
		
		printf("%d\n",sum[x2][y2]-sum[x1-1][y2]-sum[x2][y1-1]+sum[x1-1][y1-1]);
		
	}
	
	
	
	
	return 0;
}
posted @   Emila  阅读(17)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示