首页 > 其他 > 详细

[BZOJ3550] [Sdoi2014]数数

时间:2018-09-06 21:57:45      阅读:136      评论:0      收藏:0      [点我收藏+]

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;
}

 

[BZOJ3550] [Sdoi2014]数数

原文:https://www.cnblogs.com/BriMon/p/9601061.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!