1. 程式人生 > >BZOJ_2901_矩陣求和_前綴和

BZOJ_2901_矩陣求和_前綴和

static out define algorithm stdin \n string rap simple

BZOJ_2901_矩陣求和_前綴和

Description

給出兩個n*n的矩陣,m次詢問它們的積中給定子矩陣的數值和。

Input

第一行兩個正整數n,m。 接下來n行,每行n個非負整數,表示第一個矩陣。 接下來n行,每行n個非負整數,表示第二個矩陣。 接下來m行,每行四個正整數a,b,c,d,表示詢問第一個矩陣與第二個矩陣的積中,以第a行第b列與第c行第d列為頂點的子矩陣中的元素和。

Output

對每次詢問,輸出一行一個整數,表示該次詢問的答案。

Sample Input

3 2
1 9 8
3 2 0
1 8 3
9 8 4
0 5 15
1 9 6
1 1 3 3
2 3 1 2

Sample Output

661
388

【數據規模和約定】
對30%的數據滿足,n <= 100。
對100%的數據滿足,n <= 2000,m <= 50000,輸入數據中矩陣元素 < 100,a,b,c,d <= n。

$\sum\limits_{i=1}^{x}\sum\limits_{j=1}^{y}\sum\limits_{k=1}^{n}A_{ik}*B_{kj}$

$=\sum\limits_{k=1}^{n}\sum\limits_{i=1}^{n}A_{ik}\sum\limits_{j=1}^{y}B_{kj}$
處理出前綴和之後每次O(n)查一遍。 代碼:
// bzoj-judger-enable-ogay
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
int sa[2050][2050],sb[2050][2050],n,m;
char buf[100000],*p1=buf,*p2=buf;
#define nc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++)
__attribute__((optimize("-O998244353")))int rd() {
	int x=0; char s=nc();
	while(s<‘0‘||s>‘9‘) s=nc();
	while(s>=‘0‘&&s<=‘9‘) x=(x<<3)+(x<<1)+s-‘0‘,s=nc();
	return x;
}
char pbuf[100000],*pp=pbuf;
__attribute__((optimize("-O998244353")))void push(const char ch) {
	if(pp-pbuf==100000) fwrite(pbuf,1,100000,stdout),pp=pbuf;
	*pp++=ch;
}
__attribute__((optimize("-O998244353")))void write(ll x) {
	static int sta[70];
	int top=0;
	do{sta[++top]=x%10,x/=10;}while(x);
	while(top) push(sta[top--]+‘0‘);
	push(‘\n‘);
}
__attribute__((optimize("-O998244353")))ll qu(int x,int y,int z,int w) {
	int i;
	ll re=0;
	for(i=1;i<=n;i++) re+=ll(sa[z][i]-sa[x-1][i])*(sb[i][w]-sb[i][y-1]);
	return re;
}
__attribute__((optimize("-O998244353")))int main() {
	n=rd(); m=rd();
	register int i,j,x;
	for(i=1;i<=n;i++) {
		for(j=1;j<=n;j++) {
			x=rd(); sa[i][j]=sa[i-1][j]+x;
		}
	}
	for(i=1;i<=n;i++) {
		for(j=1;j<=n;j++) {
			x=rd(); sb[i][j]=sb[i][j-1]+x;
		}
	}
	int y,z,w;
	while(m--) {
		x=rd(); y=rd(); z=rd(); w=rd();
		if(x>z) swap(x,z); if(y>w) swap(y,w);
		write(qu(x,y,z,w));
	}
	fwrite(pbuf,1,pp-pbuf,stdout);
}

BZOJ_2901_矩陣求和_前綴和