UVA10328

题目链接: http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=1269

题意:抛n次骰子,求至少出现 连续k次 正面的有多少次

分析:动态规划,dp[i][j]表示第i次为出现j (j = 0为正面,j = 1为反面 ),出现至多连续u次正面,v次反面的所有可能情况

至少出现k次正面,就是连续至多出现n次正面,n次反面,-至多出现k-1次正面,n次反面

dp[i][j] = dp[i - 1][0] + dp[i - 1][1](i <= u)

dp[i][j] = dp[i - 1][0] + dp[i - 1][1] - 1(i = u + 1)

dp[i][j] = dp[i - 1][0] + dp[i - 1][1] - dp[i - u - 1][1](i > u + 1)

注意没有取模,只能高精度

代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int L = 110;
string dp[110][2];
int n,k;
string add(string a,string b)//只限两个非负整数相加
{
    string ans;
    int na[L]={0},nb[L]={0};
    int la=a.size(),lb=b.size();
    for(int i=0;i<la;i++) na[la-1-i]=a[i]-'0';
    for(int i=0;i<lb;i++) nb[lb-1-i]=b[i]-'0';
    int lmax=la>lb?la:lb;
    for(int i=0;i<lmax;i++) na[i]+=nb[i],na[i+1]+=na[i]/10,na[i]%=10;
    if(na[lmax]) lmax++;
    for(int i=lmax-1;i>=0;i--) ans+=na[i]+'0';
    return ans;
}
 string sub(string a,string b)//只限大的非负整数减小的非负整数
{
    string ans;
    int na[L]={0},nb[L]={0};
    int la=a.size(),lb=b.size();
    for(int i=0;i<la;i++) na[la-1-i]=a[i]-'0';
    for(int i=0;i<lb;i++) nb[lb-1-i]=b[i]-'0';
    int lmax=la>lb?la:lb;
    for(int i=0;i<lmax;i++)
    {
        na[i]-=nb[i];
        if(na[i]<0) na[i]+=10,na[i+1]--;
    }
    while(!na[--lmax]&&lmax>0)  ;lmax++;
    for(int i=lmax-1;i>=0;i--) ans+=na[i]+'0';
    return ans;
}
string work(int u)//至多u个H,至多k个T,求
{
    dp[0][0] = "1";
    dp[0][1] = "0";
    for(int i = 1; i <= n; i++)//0 H 1 T
    {
        dp[i][1] = add(dp[i - 1][0],dp[i - 1][1]);
        if(i <= u)
        {
            dp[i][0] = add(dp[i - 1][0],dp[i - 1][1]);
        }
        else if(i == u + 1)
        {
            dp[i][0] = add(dp[i - 1][0],dp[i - 1][1]);
            dp[i][0] = sub(dp[i][0],"1");
        }
        else
        {
            dp[i][0] = add(dp[i - 1][0],dp[i - 1][1]);
            dp[i][0] = sub(dp[i][0],dp[i - u - 1][1]);
        }
    }
    return add(dp[n][0],dp[n][1]);
}
int main()
{

    //至少k个H,至多n个H-至多k-1个H.至多n个T
    while(~scanf("%d%d",&n,&k))
    {
        string res;
        res = sub(work(n),work(k - 1));
        cout<<res<<endl;
    }
    return 0;
}

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值