Luogu1736 創意喫魚法

DP。

\(f(i,j)\)爲右下角是\((i,j)\)的能喫到的最多的魚,\(line(i,j)\)\(col(i,j)\)分別爲向左/右和向上能擴展到的全是0的最長的長度。

容易寫出轉移方程:

\(a[i][j]=0\)\(line[i][j]=line[i][j-1]+1, col[i][j]=col[i-1][j]+1\)

\(a[i][j]=1\)時求一遍\(f[i][j]=min(f[i-1][j-1], min(line[i][j-1], col[i-1][j]))+1\)

DP兩遍,分別是左上-右下和左下-右上qwq

CODE:

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int a[2501][2501], line[2501][2501], col[2501][2501], f[2501][2501];
int n, m, ans;
int main(){
	scanf("%d%d", &n, &m);
	for(int i=1; i<=n; i++){
		for(int j=1; j<=m; j++){
			scanf("%d", &a[i][j]);
			if(!a[i][j]){
				line[i][j]=line[i][j-1]+1;
				col[i][j]=col[i-1][j]+1;
			}else {
				f[i][j]=min(f[i-1][j-1], min(line[i][j-1], col[i-1][j]))+1;
			}
			ans=max(ans, f[i][j]);
		}
	}
	memset(f, 0, sizeof f);
	memset(line, 0, sizeof line);
	memset(col, 0, sizeof col);
	for(int i=1; i<=n; i++)
		for(int j=m; j>0; j--){
			if(!a[i][j]){
				line[i][j]=line[i][j+1]+1;
				col[i][j]=col[i-1][j]+1;
			}else {
				f[i][j]=min(f[i-1][j+1], min(line[i][j+1], col[i-1][j]))+1;
			}
			ans=max(ans, f[i][j]);
		}
	printf("%d", ans);
	return 0;
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章