• [BZOJ3550] [Sdoi2014]数数


    Description

    我们称一个正整数N是幸运数,当且仅当它的十进制表示中不包含数字串集合S中任意一个元素作为其子串。例如当S=(22,333,0233)时,233是幸运数,2333、20233、3223不是幸运数。
        给定N和S,计算不大于N的幸运数个数。

    Input


        输入的第一行包含整数N。
        接下来一行一个整数M,表示S中元素的数量。
        接下来M行,每行一个数字串,表示S中的一个元素。

    Output

        输出一行一个整数,表示答案模109+7的值。

    Sample Input

    20
    3
    2
    3
    14

    Sample Output

    14

    HINT

     下表中l表示N的长度,L表示S中所有串长度之和。

    1 < =l < =1200 , 1 < =M < =100 ,1 < =L < =1500


    在AC自动机上跑DP。

    设$large f[0/1][i][j]$表示填到第i位,匹配到自动机上第j个节点, 是否有限制的方案数。

    然后$large f[0][i][j]$是可以转移到$large f[0][i+1][nxt[j][k]]$的。

    $large f[1][i][j]$在k不等于这一位的时候可以转移到$large f[0][i+1][nxt[j][k]]$,

    在等于这一位的时候转移到$large f[1][i+1][nxt[j][k]]$。

    要特判一下匹配到根节点时,如果正在填第一位,只能从$large [1, n[1]]$中选择数转移,和上面类似,

    如果不是在填第1位,则可以直接转移到$large f[0][...][...]$。

    注意如果一个节点是一个单词的结尾就不转移,自动机上的表示要沿着fail指针传递。


    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <string>
    #include <queue>
    using namespace std;
    #define mod 1000000007
    #define reg register
    char n[1205];
    int m;
    
    int cnt;
    int nxt[1505][27], end[1505], fail[1505];
    int f[2][1205][1505];//是否有限制,正在填第i个位置,匹配到第j个点. 
    int ans;
    
    inline void Ins(string s)
    {
        int now = 0;
        int len = s.length();
        for (reg int i = 0 ; i < len ; i ++)
            now = nxt[now][s[i]-'0'] > 0 ? nxt[now][s[i]-'0'] : (nxt[now][s[i]-'0'] = ++cnt);
        end[now] = 1;
    }
    
    inline void AC_Match()
    {
        queue <int> q;
        for (reg int i = 0 ; i <= 9 ; i ++) 
            if (nxt[0][i]) q.push(nxt[0][i]);
        while(!q.empty())
        {
            int x = q.front();q.pop();
            for (reg int i = 0 ; i <= 9 ; i ++)
            {
                if (nxt[x][i]) fail[nxt[x][i]] = nxt[fail[x]][i], q.push(nxt[x][i]), end[nxt[x][i]] |= end[nxt[fail[x]][i]];
                else nxt[x][i] = nxt[fail[x]][i];
            }
        }
    }
    
    signed main()
    {
        scanf("%s", n + 1);
        scanf("%d", &m);
        for (reg int i = 1 ; i <= m ; i ++)
        {
            string x;
            cin >> x;
            Ins(x);
        }
        AC_Match(); 
        int len = strlen(n + 1);
        for (reg int i = 0 ; i < len ; i ++)
        {
            for (reg int j = 0 ; j <= cnt ; j ++)
            {
                if (f[0][i][j]) {
                    for (reg int k = 0 ; k <= 9 ; k ++)
                        if (!end[nxt[j][k]]) (f[0][i+1][nxt[j][k]] += f[0][i][j]) %= mod;
                }
                if (f[1][i][j]) {
                    int up = n[i+1] - '0';
                    for (reg int k = 0 ; k < up ; k ++)
                        if (!end[nxt[j][k]]) (f[0][i+1][nxt[j][k]] += f[1][i][j]) %= mod;
                    if (!end[nxt[j][up]]) (f[1][i+1][nxt[j][up]] += f[1][i][j]) %= mod;
                }
                if (j == 0) 
                {
                    if (i == 0) {
                        int up = n[i+1] - '0';
                        for (reg int k = 1 ; k < up ; k ++)
                            if (!end[nxt[j][k]]) (f[0][i+1][nxt[j][k]] += 1) %= mod;
                        if (!end[nxt[j][up]]) (f[1][i+1][nxt[j][up]] += 1) %= mod;
                    } else {
                        for (reg int k = 1 ; k <= 9 ; k ++)
                            if (!end[nxt[j][k]]) (f[0][i+1][nxt[j][k]] += 1) %= mod;
                    }
                }
            }
        }
        for (reg int i = 0 ; i <= cnt ; i ++)
            (ans += (f[0][len][i] + f[1][len][i]) % mod) %= mod;
        cout << ans << endl;
        return 0;
    }
  • 相关阅读:
    【操作系统】主存空间的分配和回收
    学术诚信与职业道德
    读《构建之法》第 8、9、10 章有感
    操作系统第三次作业
    sprint
    软件工程学期总结
    实验四 主存空间的分配和回收
    第二个冲刺
    Scrum项目6.0 和8910章读后感
    Spring 计划 7.0
  • 原文地址:https://www.cnblogs.com/BriMon/p/9601061.html
Copyright © 2020-2023  润新知