bzoj [Sdoi2014]数数 AC自动机上dp
[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
3
2
3
14
Sample Output
HINT
下表中l表示N的长度,L表示S中所有串长度之和。
1 < =l < =1200 , 1 < =M < =100 ,1 < =L < =1500
Source
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的更多相关文章
- 【Luogu】P3311数数(AC自动机上DP)
题目链接 蒟蒻今天终于学会了AC自动机,感觉很稳 (后一句愚人节快乐) 这题开一个f[i][j][k]表示有没有受到限制,正在枚举第j位,来到了AC自动机的第k个节点 的方案数 随后可以刷表更新 注意 ...
- 【洛谷4045】[JSOI2009] 密码(状压+AC自动机上DP)
点此看题面 大致题意: 给你\(n\)个字符串,问你有多少个长度为\(L\)的字符串,使得这些字符串都是它的子串.若个数不大于\(42\),按字典序输出所有方案. 状压 显然,由于\(n\)很小,我们 ...
- Passwords Gym - 101174E (AC自动机上DP)
Problem E: Passwords \[ Time Limit: 1 s \quad Memory Limit: 256 MiB \] 题意 给出两个正整数\(A,B\),再给出\(n\)个字符 ...
- URAL 1158 AC自动机上的简单DP+大数
题目大意 在一种语言中的字母表中有N(N<=50)个字母,每个单词都由M(M<=50)个字母构成,因此,一共可以形成N^M个单词.但是有P(P<=10)个串是被禁止的,也就是说,任何 ...
- bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)
bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...
- bzoj 2553 [BeiJing2011]禁忌——AC自动机+概率DP+矩阵
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2553 看了题解才会…… 首先,给定一个串,最好的划分方式是按禁忌串出现的右端点排序,遇到能填 ...
- 【JZOJ3624】【SDOI2014】数数(count) AC自动机+数位dp
题面 100 容易想到使用AC自动机来处理禁忌子串的问题: 然后在自动机上数位dp,具体是: \(f_{i,j,0/1}\)表示填了\(i\)位,当前在自动机的第\(j\)个结点上,\(0\)表示当前 ...
- 【HDU3530】 [Sdoi2014]数数 (AC自动机+数位DP)
3530: [Sdoi2014]数数 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 682 Solved: 364 Description 我们称一 ...
- BZOJ 1030 文本生成器 | 在AC自动机上跑DP
题目: http://www.lydsy.com/JudgeOnline/problem.php?id=1030 题解: 鸽 #include<cstdio> #include<al ...
随机推荐
- 银行卡验证API
一.银联开放平台 https://open.unionpay.com/tjweb/api/detail?apiSvcId=21 应用场景 综合数据服务平台是银联为接入商户提供的综合数据认证服务接口,目 ...
- php红包算法函数[优化]
php红包算法 <?php header("Content-Type: text/html;charset=utf-8");//输出不乱码,你懂的 $total=10000; ...
- js三目运算符执行多个条件
三元运算符的结果语句可以执行多个操作,每个操作用逗号分隔就可以,例子如下: var a=1: a>5?(alert(1),alert(2)):(alert(3),alert(4))
- jira安装说明
阅读目录 1.1 jira说明 1.2 安装配置jira 1.3 web界面访问 1.4 创建第一个项目 1.5 参考文献 回到顶部 1.1 jira说明 JIRA是Atlassian公司出品的项目与 ...
- 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 ...
- shell参数 传递
$# 是传给脚本的参数个数 $0 是脚本本身的名字 $1 是传递给该shell脚本的第一个参数 $2 是传递给该shell脚本的第二个参数 $@ 是传给脚本的所有参数的列表 $* 是以一个单字符串显示 ...
- 斐波那契数列(Fibonacci) iOS
斐波那契数列Fibonacci 斐波那契数列指的是这样一个数列 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233,377,610,987,1597,2 ...
- Git——1.简介
关于版本控制 Git基础 安装Git 初始运行Git前的配置 获取帮助 关于版本控制 版本控制(VCS)是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统. 本地版本控制系统 大多都 ...
- ES6常用片段
promise: --在return里面: methods:{ getSellData(){ return axios.get('/api/seller').then((res=>{ retur ...
- Python全栈工程师(异常(高级)、运算符重载)
ParisGabriel 每天坚持手写 一天一篇 决定坚持几年 为了梦想为了信仰 开局一张图 Python人工智能从入门到精通 对象的属性管理函数: getattr ...