zoukankan      html  css  js  c++  java
  • BZOJ3530: [Sdoi2014]数数

    【传送门:BZOJ3530


    简要题意:

      给出m个数,要求求出不含任意一个数的长度至多为n的数的数量


    题解:

      这道题比文本生成器要复杂一点,就是因为它的位数不确定,且不可以含有前导0

      所以我们用数位DP的思想来做

      设f[i][j][k]为当前长度为i且走到AC自动机的第j个点时,如果k=0,则表示仍未到顶格,k=1则表示到了顶格,这种状态下的数的数量

      只要在走的时候判断前导零的情况就可以了


    参考代码:

    #include<cstdio>
    #include<cstring>
    #include<cstdlib>
    #include<algorithm>
    #include<cmath>
    using namespace std;
    int Mod=1000000007;
    struct trie
    {
        int c[11],s,fail;
    }t[2100];int tot,root;
    void clean(int x)
    {
        memset(t[x].c,-1,sizeof(t[x].c));
        t[x].s=0;
    }
    char st[2100];
    void bt()
    {
        int x=0,len=strlen(st+1);
        for(int i=1;i<=len;i++)
        {
            int y=st[i]-'0';
            if(t[x].c[y]==-1) clean(++tot),t[x].c[y]=tot;
            x=t[x].c[y];
        }
        t[x].s=1;
    }
    int list[2100];
    void bfs()
    {
        list[1]=0;
        int head=1,tail=1;
        while(head<=tail)
        {
            int x=list[head];
            for(int i=0;i<=9;i++)
            {
                int son=t[x].c[i];
                if(son==-1) continue;
                if(x==0) t[son].fail=0;
                else
                {
                    int j=t[x].fail;
                    while(j!=0&&t[j].c[i]==-1) j=t[j].fail;
                    t[son].fail=max(0,t[j].c[i]);
                    if(t[t[son].fail].s==1) t[son].s=1;
                }
                list[++tail]=son;
            }
            head++;
        }
    }
    int a[2100];
    int f[2100][2100][2];
    int main()
    {
        scanf("%s",st+1);
        int n=strlen(st+1);
        for(int i=1;i<=n;i++) a[i]=st[i]-'0';
        int m;
        scanf("%d",&m);
        tot=0;clean(0);
        for(int i=1;i<=m;i++)
        {
            scanf("%s",st+1);
            bt();
        }
        bfs();
        for(int i=1;i<=a[1];i++)
        {
            int x=t[0].c[i];
            if(x==-1) x=0;
            if(i<a[1]) f[1][x][0]++;
            if(i==a[1]) f[1][x][1]++;
        }
        for(int i=2;i<=n;i++)
        {
            for(int j=1;j<=9;j++)
            {
                int x=t[0].c[j];
                if(x==-1) x=0;
                f[i][x][0]++;
            }
            for(int j=0;j<=tot;j++)
            {
                if(t[j].s==0)
                {
                    for(int k=0;k<=9;k++)
                    {
                        int son=j;
                        while(son!=0&&t[son].c[k]==-1) son=t[son].fail;
                        son=t[son].c[k];
                        if(son==-1||t[son].s==0)
                        {
                            if(i==1&&k==0) continue;
                            if(son==-1) son=0;
                            f[i][son][0]=(f[i][son][0]+f[i-1][j][0])%Mod;
                            if(k<a[i]) f[i][son][0]=(f[i][son][0]+f[i-1][j][1])%Mod;
                            if(k==a[i]) f[i][son][1]=(f[i][son][1]+f[i-1][j][1])%Mod;
                        }
                    }
                }
            }
        }
        int ans=0;
        for(int i=0;i<=tot;i++) if(t[i].s==0) ans=(ans+f[n][i][0]+f[n][i][1])%Mod;
        printf("%d
    ",ans);
        return 0;
    }

     

  • 相关阅读:
    一个简单的随机数生成算法实现(C++)
    gabor 滤波的c++实现与该类得使用简介
    嵌入式软件的覆盖测试
    scanf()函数用法小结(转载)
    创建动态2维vector (C++)
    HDU 1086 You can Solve a Geometry Problem too
    计算几何多边形的重心
    HDU 1711 Number Sequence
    HDU 2602 Bone Collector
    计算几何基础篇
  • 原文地址:https://www.cnblogs.com/Never-mind/p/8960753.html
Copyright © 2011-2022 走看看