[Sdoi2014]数数

Time Limit: 10 Sec  Memory Limit: 512 MB
Submit: 1264  Solved: 636
[Submit][Status][Discuss]

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

Source

Round 1 day 1

AC自动机建好后f[i][j][k]表示i这个点j长度k=0表示小于,1表示等于

这样转移很好转移 IL的复杂度随便跑,

虑到不能有前导零,所以填每一位的时候可以把根节点单独转移一次1~9

 #pragma GCC optimzie(2)
#pragma G++ optimize(2)
#include<cstring>
#include<cmath>
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<queue> #define ll long long
#define N 2007
#define mod 1000000007
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f=-;ch=getchar();}
while(isdigit(ch)){x=(x<<)+(x<<)+ch-'';ch=getchar();}
return x*f;
} int m,len,l;
ll ans;
struct Node
{
int c[],suf;
bool flag;
}trie[N];
int cnt=,rt=;
ll f[][][];
char ch[N],n[N]; void insert()
{
int now=rt;
for (int i=;i<=len;i++)
if(trie[now].c[ch[i]-''])now=trie[now].c[ch[i]-''];
else
{
trie[now].c[ch[i]-'']=++cnt;
now=cnt;
}
trie[now].flag=true;
}
void make_AC()
{
for (int i=;i<;i++)
trie[].c[i]=;
queue<int>q;q.push();
trie[].suf=;
while(!q.empty())
{
int u=q.front();q.pop();
for (int i=;i<;i++)
if(trie[u].c[i])
{
trie[trie[u].c[i]].suf=trie[trie[u].suf].c[i];
if(trie[trie[trie[u].c[i]].suf].flag)trie[trie[u].c[i]].flag=true;
q.push(trie[u].c[i]);
}
else trie[u].c[i]=trie[trie[u].suf].c[i];
}
}
void solve()
{
/*f[0][1][1]=1;
for (int i=0;i<l;i++)
for (int j=1;j<=cnt;j++)
{
if (f[i][j][1]&&!trie[j].flag)
{
for (int k=0;k<10;k++)
if (n[i+1]-'0'>k)(f[i+1][trie[j].c[k]][0]+=f[i][j][1])%=mod;
else if(n[i+1]-'0'==k)(f[i+1][trie[j].c[k]][1]+=f[i][j][1])%=mod;
}
if(f[i][j][0]&&!trie[j].flag)
{
for (int k=0;k<10;k++)
(f[i+1][trie[j].c[k]][0]+=f[i][j][0])%=mod;
}
}*/
for (int i=;i<=l;i++)
{
for (int z=,v=trie[].c[z];z<=;v=trie[].c[++z])
{
if(i!=||z<n[i]-'')(f[i][v][]+=)%=mod;
else if(z==n[i]-'')(f[i][v][]+=)%=mod;
}
for (int j=;j<=cnt;j++)
if(!trie[j].flag)
for (int c=,v=trie[j].c[c];c<=;v=trie[j].c[++c])
{
(f[i][v][]+=f[i-][j][])%=mod;
if(c<n[i]-'')(f[i][v][]+=f[i-][j][])%=mod;
else if(c==n[i]-'')(f[i][v][]+=f[i-][j][])%=mod;
}
} for (int i=;i<=cnt;i++)
if(!trie[i].flag)(ans+=f[l][i][]+f[l][i][])%=mod;
printf("%lld",ans);
}
int main()
{
scanf("%s",n+);l=strlen(n+);
m=read();
for (int i=;i<=m;i++)
{
scanf("%s",ch+),len=strlen(ch+);
insert();
}
make_AC(),solve();
}

bzoj [Sdoi2014]数数 AC自动机上dp的更多相关文章

  1. 【Luogu】P3311数数(AC自动机上DP)

    题目链接 蒟蒻今天终于学会了AC自动机,感觉很稳 (后一句愚人节快乐) 这题开一个f[i][j][k]表示有没有受到限制,正在枚举第j位,来到了AC自动机的第k个节点 的方案数 随后可以刷表更新 注意 ...

  2. 【洛谷4045】[JSOI2009] 密码(状压+AC自动机上DP)

    点此看题面 大致题意: 给你\(n\)个字符串,问你有多少个长度为\(L\)的字符串,使得这些字符串都是它的子串.若个数不大于\(42\),按字典序输出所有方案. 状压 显然,由于\(n\)很小,我们 ...

  3. Passwords Gym - 101174E (AC自动机上DP)

    Problem E: Passwords \[ Time Limit: 1 s \quad Memory Limit: 256 MiB \] 题意 给出两个正整数\(A,B\),再给出\(n\)个字符 ...

  4. URAL 1158 AC自动机上的简单DP+大数

    题目大意 在一种语言中的字母表中有N(N<=50)个字母,每个单词都由M(M<=50)个字母构成,因此,一共可以形成N^M个单词.但是有P(P<=10)个串是被禁止的,也就是说,任何 ...

  5. bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)

    bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...

  6. bzoj 2553 [BeiJing2011]禁忌——AC自动机+概率DP+矩阵

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2553 看了题解才会…… 首先,给定一个串,最好的划分方式是按禁忌串出现的右端点排序,遇到能填 ...

  7. 【JZOJ3624】【SDOI2014】数数(count) AC自动机+数位dp

    题面 100 容易想到使用AC自动机来处理禁忌子串的问题: 然后在自动机上数位dp,具体是: \(f_{i,j,0/1}\)表示填了\(i\)位,当前在自动机的第\(j\)个结点上,\(0\)表示当前 ...

  8. 【HDU3530】 [Sdoi2014]数数 (AC自动机+数位DP)

    3530: [Sdoi2014]数数 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 682  Solved: 364 Description 我们称一 ...

  9. BZOJ 1030 文本生成器 | 在AC自动机上跑DP

    题目: http://www.lydsy.com/JudgeOnline/problem.php?id=1030 题解: 鸽 #include<cstdio> #include<al ...

随机推荐

  1. 银行卡验证API

    一.银联开放平台 https://open.unionpay.com/tjweb/api/detail?apiSvcId=21 应用场景 综合数据服务平台是银联为接入商户提供的综合数据认证服务接口,目 ...

  2. php红包算法函数[优化]

    php红包算法 <?php header("Content-Type: text/html;charset=utf-8");//输出不乱码,你懂的 $total=10000; ...

  3. js三目运算符执行多个条件

    三元运算符的结果语句可以执行多个操作,每个操作用逗号分隔就可以,例子如下: var a=1: a>5?(alert(1),alert(2)):(alert(3),alert(4))

  4. jira安装说明

    阅读目录 1.1 jira说明 1.2 安装配置jira 1.3 web界面访问 1.4 创建第一个项目 1.5 参考文献 回到顶部 1.1 jira说明 JIRA是Atlassian公司出品的项目与 ...

  5. POJ:2429-GCD & LCM Inverse(素数判断神题)(Millar-Rabin素性判断和Pollard-rho因子分解)

    原题链接:http://poj.org/problem?id=2429 GCD & LCM Inverse Time Limit: 2000MS Memory Limit: 65536K To ...

  6. shell参数 传递

    $# 是传给脚本的参数个数 $0 是脚本本身的名字 $1 是传递给该shell脚本的第一个参数 $2 是传递给该shell脚本的第二个参数 $@ 是传给脚本的所有参数的列表 $* 是以一个单字符串显示 ...

  7. 斐波那契数列(Fibonacci) iOS

    斐波那契数列Fibonacci 斐波那契数列指的是这样一个数列 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233,377,610,987,1597,2 ...

  8. Git——1.简介

    关于版本控制 Git基础 安装Git 初始运行Git前的配置 获取帮助 关于版本控制 版本控制(VCS)是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统. 本地版本控制系统 大多都 ...

  9. ES6常用片段

    promise: --在return里面: methods:{ getSellData(){ return axios.get('/api/seller').then((res=>{ retur ...

  10. Python全栈工程师(异常(高级)、运算符重载)

    ParisGabriel              每天坚持手写  一天一篇  决定坚持几年 为了梦想为了信仰    开局一张图 Python人工智能从入门到精通 对象的属性管理函数: getattr ...