zoukankan      html  css  js  c++  java
  • hdu3886(数位dp)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=3886

    题意:给一定区间[A,B],一串由/,\,-组成的符号串。求满足符号串的数字个数。

    •/表示数字从左到右递增
    •表示数字从左到右递减
    •-表示数字从左到右相等
    分析:dp[i][j][k],表示当枚举到第i位的数,匹配str[j],前一位是k,满足要求的数字个数.
     
    #include <cstdio>
    #include <cstring>
    #include <string>
    #include <cmath>
    #include <iostream>
    #include <algorithm>
    #include <queue>
    #include <cstdlib>
    #include <stack>
    #include <vector>
    #include <set>
    #include <map>
    #define LL long long
    #define mod 100000000
    #define inf 0x3f3f3f3f
    #define N 100010
    #define FILL(a,b) (memset(a,b,sizeof(a)))
    #define lson l,m,rt<<1
    #define rson m+1,r,rt<<1|1
    using namespace std;
    int dp[110][110][10];
    int dig[110],len;
    char str[110],a[110],b[110];
    int judge(int i,int pre,int now)
    {
        if(str[i]=='/')return pre<now;
        else if(str[i]=='-')return pre==now;
        else if(str[i]=='\')return pre>now;
    }
    int  dfs(int pos,int cur,int pre,int limit,int fzore)
    {
        if(!pos)return cur==len;
        if(!limit&&~dp[pos][cur][pre])return dp[pos][cur][pre];
        int ed=limit?dig[pos]:9;
        int ans=0;
        for(int i=0;i<=ed;i++)
        {
            if(!fzore)
            {
                if(cur<len&&judge(cur,pre,i))
                ans+=dfs(pos-1,cur+1,i,limit&&i==ed,fzore&&!i);
                //这里为什么能往回走,因为如果pre,i满足str[cur-1],同时
                //已经知道“pre前一位,pre”也满足str[cur-1],这样还是满足要求的数
                //例如数据123455555554321是符合/-的,5之前都是/,中间都是-,后面都是。
                else if(cur&&judge(cur-1,pre,i))
                ans+=dfs(pos-1,cur,i,limit&&i==ed,fzore&&!i);
            }
            else ans+=dfs(pos-1,cur,i,limit&&i==ed,fzore&&!i);
            ans%=mod;
        }
        if(!limit)dp[pos][cur][pre]=ans;
        return ans;
    }
    int solve(char s[],bool left)
    {
        int lens=strlen(s),m=0,i=0;
        while(s[i]=='0'&&i<lens)i++;
        for(int j=lens-1;j>=i;j--)dig[++m]=s[j]-'0';
        if(left&&m)
        {
            for(int i=1;i<=m;i++)
            {
                if(dig[i])
                {
                    dig[i]--;break;
                }
                else dig[i]=9;
            }
        }
        return dfs(m,0,0,1,1);
    }
    int main()
    {
        while(scanf("%s",str)!=EOF)
        {
            len=strlen(str);
            scanf("%s%s",a,b);
            memset(dp,-1,sizeof(dp));
            printf("%08d
    ",((solve(b,0)-solve(a,1))+mod)%mod);
        }
    }
    View Code
  • 相关阅读:
    生物创新科技素养大赛小车代码
    对拍程序
    Link-Cut-Tree学习笔记
    可平面性判定,任意平面图判定(代码实现)
    强联通缩点拓扑排序去重边小技巧
    20200405~06题解
    数论总结
    20200328题解
    Dp优化总结
    20200314题解
  • 原文地址:https://www.cnblogs.com/lienus/p/4249310.html
Copyright © 2011-2022 走看看