Zoj 3543 Number String (dp) - 2011 ACM-ICPC Dalian Regional Contest Problem E

又是一道dp。比赛时以为是数学题,一直在找规律推公式。

/**

题意:
由{1,2,3}组成的一个排列132,对应一个字符串"ID",'I'表示Increase,'D'表示Decrease,
对于排列"132",因为 1 < 3 > 2,所以对应的字符串为"ID"。现在反过来,输入一个字符串,
仅包含'I','D','?'三种字符,字符串长度n(n<=1000),输出由{1...n+1}组成的排列中,满足
该字符串的个数与1000000007求余。
题解:
dp[i][j] 表示长度为i以j结尾的合法的排列个数(由1...i组成的排列)。
还要清楚一个结论:
给定一个长度为i-1的字符串,由{1,2,...,i}组成的合法排列数和由{1,2,...,j-1,j+1,...,i+1}
组成的合法排列数是相同的。
if ( s[i] == 'I' ) dp[i][j] = dp[i-1][1] + dp[i-1][2] + ... + dp[i-1][j-1];
if ( s[i] == 'D' ) dp[i][j] = dp[i-1][j] + dp[i-1][j+1] +...+ dp[i-1][i-1];
if ( s[i] == '?' ) dp[i][j] = dp[i-1][1] + dp[i-1][2] + ... + dp[i-1][i-1];


时间复杂度是O(n^3)的,但定义一个sum[i][j]利用部分求和,就转化成了O(n^2).

**/

#include <iostream>
#include <cstdio>
#include <cstring>

using namespace std;

typedef long long LL;
const int MAXN = 1000 + 5;
const LL  MOD  = 1000000007;

char s[MAXN];
LL   dp[MAXN][MAXN], sum[MAXN][MAXN];

int main()
{
    while ( scanf("%s", s+2) != EOF )
    {
        int len = strlen(s+2) + 1;
        sum[1][1] = dp[1][1] = 1;
        for (int i = 2; i <= len; i++)
        {
            for (int j = 1; j <= i; j++)
            {
                if ( s[i] == 'I' )
                    dp[i][j] = sum[i-1][j-1];
                else if ( s[i] == 'D' )
                    dp[i][j] = sum[i-1][i-1] - sum[i-1][j-1];
                else
                    dp[i][j] = sum[i-1][i-1];
                sum[i][j] = (sum[i][j-1] + dp[i][j]) % MOD;
            }
        }
        printf("%lld\n", (sum[len][len] + MOD) % MOD); // 因为sum中都保存的是求余之后的值,前面的想减可能产生负数
    }
    return 0;
}


“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
©️2022 CSDN 皮肤主题:大白 设计师:CSDN官方博客 返回首页
评论

打赏作者

morgan_xww

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值