Bzoj 1055: [HAOI2008]玩具取名 (区间DP) Bzoj 1055: [HAOI2008]玩具取名 (区间DP)

Bzoj 1055: [HAOI2008]玩具取名 (区间DP)
Bzoj 1055: [HAOI2008]玩具取名 (区间DP)

题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1055
区间动态规划和可达性DP的好题.
开始WA了几次,感到非常奇怪.
原因竟然是n被定义了char型,真是zz了.
(f[i][j][k])表示区间(i)(j)可以由(k)这个字符是否可以转变过来.
转移的时候枚举中间点转移就好了.

#include <iostream>
#include <cstring>
#include <cstdio>
const int maxN = 200 + 7;
using namespace std;
 
const char Q[] = {' ','W','I','N','G'};
 
bool f[maxN][maxN][5];
bool is_match[5][5][5];
int q[5];
char s[maxN],c[5];

inline int pd(char i){  
    if(i == Q[1]) return 1; 
    if(i == Q[2]) return 2;
    if(i == Q[3]) return 3; 
    if(i == Q[4]) return 4; 
}  
 
int main() {
    for(int i = 1;i <= 4;++ i) 
        scanf("%d",&q[i]);
    for(int i = 1;i <= 4;++ i) {
        for(int j = 1;j <= q[i];++ j) {
            scanf("%s",c);
            is_match[i][pd(c[0])][pd(c[1])] = true;
        }
    }
    scanf("%s",s + 1); 
    int n = strlen(s + 1);
    for(int i = 1;i <= n;++ i) 
        f[i][i][pd(s[i])] = true;
    for(int leg = 1;leg < n;++ leg) {
        for(int l = 1;l + leg <= n;++ l) {
        	int r = l + leg;
        	for(int k = l;k < r;++ k) {
        		for(int i = 1;i <= 4;++ i) { //枚举 l 到 r 的字符 
        			for(int j = 1;j <= 4;++ j) {// 枚举l 到 k 的字符
        				for(int a = 1;a <= 4;++ a) { //枚举k + 1到r的字符
        					if(is_match[i][j][a]) {
        						if( f[l][k][j] && f[k + 1][r][a])
        							f[l][r][i] = 1;
							}
        				}
        			}
        		}
        	}
        }
    }
    bool flag = false;
    for(int i = 1;i <= 4;++ i)
        if(f[1][n][i]) flag = true,printf("%c",Q[i]);
    if(!flag) puts("The name is wrong!");
    return 0;
}