【LOJ】#2349. 「JOI 2017/2018 决赛」团子制作

时间:2021-05-12 17:57:34

题解

有意思的一个dp,我们对G计数,发现如果不在同一条对角线上的G肯定不会互相影响,所以我们对于每一条对角线dp

dp的方式是枚举这个G以什么方式放,横着还是竖着,还是不放

代码

#include <iostream>
#include <cstdio>
#include <vector>
#include <algorithm>
#include <cmath>
#include <cstring>
#include <map>
#include <queue>
//#define ivorysi
#define pb push_back
#define space putchar(' ')
#define enter putchar('\n')
#define mp make_pair
#define pb push_back
#define fi first
#define se second
#define mo 974711
#define MAXN 100005
#define MAXM 200005
#define eps 1e-3
#define RG register
#define calc(x) __builtin_popcount(x)
#define pLI pair<int64,int>
using namespace std;
typedef long long int64;
typedef double db;
template<class T>
void read(T &res) {
res = 0;char c = getchar();T f = 1;
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 + c - '0';
c = getchar();
}
res *= f;
}
template<class T>
void out(T x) {
if(x < 0) {putchar('-');x = -x;}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
int N,M,ans,dp[3][3005];
char s[3005][3005];
bool vis[3005][3005],has[3005][3005];
void Solve() {
read(N);read(M);
for(int i = 1 ; i <= N ; ++i) scanf("%s",s[i] + 1);
for(int h = 2 ; h <= N + M ; ++h) {
memset(dp,0,sizeof(dp));
int tmp = 0;
for(int i = max(1,h - M) ; i <= min(h - 1,N) ; ++i) {
int j = h - i;
dp[0][i] = max(max(dp[0][i - 1],dp[1][i - 1]),dp[2][i - 1]);
if(s[i][j] == 'G') {
if(s[i - 1][j] == 'R' && s[i + 1][j] == 'W') {
dp[1][i] = max(dp[0][i - 1],dp[1][i - 1]) + 1;
}
if(s[i][j - 1] == 'R' && s[i][j + 1] == 'W') {
dp[2][i] = max(dp[0][i - 1],dp[2][i - 1]) + 1;
}
}
tmp = max(tmp,max(max(dp[0][i],dp[1][i]),dp[2][i]));
}
ans += tmp;
}
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Solve();
return 0;
}