uva10401 - Injured Queen Problem(简单动归)

题意:

受伤的皇后只能攻击它这一列的,和它周围的九个格子。

给定一个字符串,如果第i个字符是?则表示皇后能放在第i列的任意位置,如果不是?则表示它指定了必须放在哪一行,问有几种放法。

思路:

这种格子题目类似于小明回家,有几条路径可以选择一样,只不过初始点和结束状态不怎么一样而已。

状态:dp[i, j]表示(i, j)坐标最多有几种放法;

状态转移:dp[i, j]+=dp[k, j-1],(k取决于题目条件)。

#include <cstdio>
#include <cstring>
#define M 20
long long dp[M][M];
char str[M];
int trans(int x)
{
    char ch = str[x];
    if(ch>='0'&&ch<='9') return ch-'0';
    else return ch-'A'+10;
}
int main ()
{
    int cur;
    while(~scanf("%s",str))
    {
        memset(dp,0,sizeof(dp));
        int len = strlen(str);
        if(str[0]=='?')
        for(int i = 1; i <= len; i++) dp[i][1] = 1;
        else
        {
            cur = trans(0);
            dp[cur][1] = 1;
        }
        for(int i = 2; i <= len; i++)
        {
            if(str[i-1]=='?')
            {
                for(int j = 1; j <= len; j++)
                    for(int k = 1; k <= len; k++) if(k!=j&&k!=j-1&&k!=j+1)
                    dp[j][i] += dp[k][i-1];
            }
            else
            {
                cur = trans(i-1);
                for(int k = 1; k <= len; k++) if(k!=cur&&k!=cur-1&&k!=cur+1)
                dp[cur][i] += dp[k][i-1];
            }
        }
        long long ans = 0;
        for(int i = 1; i <= len; i++) ans+=dp[i][len];
        printf("%lld\n",ans);
    }
    return 0;
}

你可能感兴趣的:(uva10401 - Injured Queen Problem(简单动归))