码迷,mamicode.com
首页 > 其他好文 > 详细

FZU2126:消去游戏(DP)

时间:2014-12-17 12:56:48      阅读:121      评论:0      收藏:0      [点我收藏+]

标签:dp   fzu   

Problem Description

S最近在玩一种游戏。这种游戏的规则是一个一个地往一个栈里放有颜色的球,当栈顶连续k个球颜色相同时,这k个球立刻同时消失。现在S已经往栈里放了n个球,他想知道再放m个球,然后使得栈里的球都被消去的放法有多少种。两种放法不同是指存在放的第i个球这两种放法放的球的颜色不同。由于方法数可能很多,将答案mod 1000000007。

bubuko.com,布布扣 Input

输入包含多组数据。输入数据的第一行为四个整数n,m,h,k(0<=n,m,h<=1000,2<=k<=1000),表示已经放了n个球,有h种不同颜色的球,若栈顶出现连续k个球颜色相同则这k个球同时消失,问再放m个球,使得最后栈里的球都被消去的放法数。第二行从左往右依次输入n个整数,范围为1到h,表示刚开始往栈里放的球的颜色,放入顺序与输入顺序相同,数据保证已经放入的n个球不会存在连续k个球颜色相同。答案对1000000007取余。

bubuko.com,布布扣 Output

输出一行一个整数M,表示对1000000007取余后的放法数。

bubuko.com,布布扣 Sample Input

3 6 3 3 1 2 20 6 2 3

bubuko.com,布布扣 Sample Output

98

dp[i][j]代表还需要放i个球,还有j个球需要消去的情况

#include<stdio.h>
#include<algorithm>
#include<string.h>
using namespace std;
const int mod = 1000000007;
int n,m,h,k;
int cur,pre,tot;
long long dp[1005][1005];
int main()
{
    int i,j;
    while(~scanf("%d%d%d%d",&n,&m,&h,&k))
    {
        memset(dp,0,sizeof(dp));
        pre = cur = tot = 0;
        for(i = 0; i<n; i++)
        {
            scanf("%d",&cur);
            if(cur != pre)//不等于前面,还需要k-1个球才能消去
                tot+=k-1;
            else//来一个相等的就减去1
                tot--;
            pre = cur;
        }
        dp[m][tot] = 1;//直接按现在所存在的数字消去的方法只有一种
        for(i = m; i>0; i--)
        {
            for(j = i; j>=0; j--)
            {
                if(i-1>=j+k-1)//如果末尾放一个与栈末尾的求不同的球,那么相应的情况要多加k-1个球
                    dp[i-1][j+k-1] = (dp[i-1][j+k-1]+dp[i][j]*(j?(h-1):h))%mod;
                if(j-1<0)
                    break;
                dp[i-1][j-1] = (dp[i-1][j-1]+dp[i][j])%mod;//根据前面可以知道,我在末尾放一个与栈尾相同的球,那么我还需要放的个数必然是减去1的
            }
        }
        printf("%lld\n",dp[0][0]);
    }

    return 0;
}


FZU2126:消去游戏(DP)

标签:dp   fzu   

原文地址:http://blog.csdn.net/libin56842/article/details/41978389

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!