BZOJ3530:[SDOI2014]数数(AC自动机,数位DP)
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
Solution
一个挺简单的一个题……建出来$AC$自动机然后在上面直接跑数位$DP$就好了。只不过有点小地方需要注意。
用$DFS(zero,lim,pos,now)$是否有前导0,是否卡上界,第$pos$位,自动机上第$now$个点。
为什么要存前导0呢?我们可以发现有这么一个例子:
10
1
01
这个跑出来应该是10,然而不记前导零特判一下会跑出来9。这是因为当幸运串为01的时候我们会忽略前导0,所以是合法的。
只需要在$DFS$的时候特判一下,如果有前导0,且幸运数这一位选0,且$now$还在根节点,就让$now$停在根节点就好了。
建立AC自动机的时候,如果某个节点能够沿着fail指针跳到单词节点,那么这个节点也应当禁止通过……
自测一时爽
Code
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
#define N (1509)
#define MOD (1000000007)
using namespace std; int sz,Son[N][],Fail[N],End[N];
int cnt,m,x,f[N][N],a[N],spc[N];
char n[N],s[N];
queue<int>q; void Insert(char s[])
{
int now=;
for (int i=,l=strlen(s); i<l; ++i)
{
int c=s[i]-'';
if (!Son[now][c]) Son[now][c]=++sz;
now=Son[now][c];
}
End[now]++;
} void Build_Fail()
{
for (int i=; i<=; ++i)
if (Son[][i]) q.push(Son[][i]);
while (!q.empty())
{
int now=q.front(); q.pop();
for (int i=; i<=; ++i)
{
if (!Son[now][i])
{
Son[now][i]=Son[Fail[now]][i];
continue;
}
Fail[Son[now][i]]=Son[Fail[now]][i];
q.push(Son[now][i]);
}
}
} int DFS(int zero,int lim,int pos,int now)
{
if (pos==) return ;
if (!zero && !lim && f[pos][now]!=-) return f[pos][now];
f[pos][now]=;
int up=lim?n[pos]-'':;
for (int i=; i<=up; ++i)
if (!End[Son[now][i]])
{
if (zero && !i && !now) (f[pos][now]+=DFS(zero,lim&&==up,pos-,))%=MOD;
else
{
int flag=,t=Son[now][i];
while (t)
{
if (End[t]) {flag=; break;}
t=Fail[t];
}
if (!flag) continue;
(f[pos][now]+=DFS(zero&&!i,lim&&i==up,pos-,Son[now][i]))%=MOD;
}
}
return f[pos][now];
} int main()
{
memset(f,-,sizeof(f));
scanf("%s%d",n+,&m); cnt=strlen(n+);
for (int i=; i<=m; ++i)
scanf("%s",s),Insert(s);
Build_Fail();
for (int i=,j=cnt; i<j; ++i,--j)
swap(n[i],n[j]);
printf("%d\n",DFS(,,cnt,)-);
}
BZOJ3530:[SDOI2014]数数(AC自动机,数位DP)的更多相关文章
- 【HDU3530】  [Sdoi2014]数数 (AC自动机+数位DP)
		3530: [Sdoi2014]数数 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 682 Solved: 364 Description 我们称一 ... 
- 【JZOJ3624】【SDOI2014】数数(count) AC自动机+数位dp
		题面 100 容易想到使用AC自动机来处理禁忌子串的问题: 然后在自动机上数位dp,具体是: \(f_{i,j,0/1}\)表示填了\(i\)位,当前在自动机的第\(j\)个结点上,\(0\)表示当前 ... 
- 【bzoj3530】[Sdoi2014]数数  AC自动机+数位dp
		题目描述 我们称一个正整数N是幸运数,当且仅当它的十进制表示中不包含数字串集合S中任意一个元素作为其子串.例如当S=(22,333,0233)时,233是幸运数,2333.20233.3223不是幸运 ... 
- BZOJ 3530 [SDOI2014]数数 (Trie图/AC自动机+数位DP)
		题目大意:略 裸的AC自动机+数位DP吧... 定义f[i][x][0/1]表示已经匹配到了第i位,当前位置是x,0表示没到上限,1到上限,此时数是数量 然而会出现虚拟前导零,即前几位没有数字的情况, ... 
- BZOJ3530[Sdoi2014]数数——AC自动机+数位DP
		题目描述 我们称一个正整数N是幸运数,当且仅当它的十进制表示中不包含数字串集合S中任意一个元素作为其子串.例如当S=(22,333,0233)时,233是幸运数,2333.20233.3223不是幸运 ... 
- BZOJ 3530: [Sdoi2014]数数 [AC自动机 数位DP]
		3530: [Sdoi2014]数数 题意:\(\le N\)的不含模式串的数字有多少个,\(n=|N| \le 1200\) 考虑数位DP 对于长度\(\le n\)的,普通套路DP\(g[i][j ... 
- [SDOI2014]数数 --- AC自动机 + 数位DP
		[SDOI2014]数数 题目描述: 我们称一个正整数N是幸运数,当且仅当它的十进制表示中不包含数字串集合S中任意一个元素作为其子串. 例如当S=(22,333,0233)时,233是幸运数,2333 ... 
- P3311 [SDOI2014]数数 AC自动机+数位DP
		题意 给定一个正整数N和n个模式串,问不大于N的数字中有多少个不包含任意模式串,输出对\(1e^9+7\)取模后的答案. 解题思路 把所有模式串都加入AC自动机,然后跑数位DP就好了.需要注意的是,这 ... 
- HDU-4518 吉哥系列故事——最终数 AC自动机+数位DP
		题意:如果一个数中的某一段是长度大于2的菲波那契数,那么这个数就被定义为F数,前几个F数是13,21,34,55......将这些数字进行编号,a1 = 13, a2 = 21.现给定一个数n,输出和 ... 
随机推荐
- 【转】详解spring 每个jar的作用
			spring.jar 是包含有完整发布模块的单个jar 包.但是不包括mock.jar, aspects.jar, spring-portlet.jar, and spring-hibernate2. ... 
- Differences between write through and write back
			https://stackoverflow.com/questions/27087912/write-back-vs-write-through 
- 关于shader的学习
			教程地址:https://onevcat.com/2013/07/shader-tutorial-1 因为想做一些特效,所以想稍微了解一下shader的代码,一下是一些笔记 // Upgrade NO ... 
- Spring是什么、spring容器、Spring三大核心思想DI(依赖注入)、IOC(控制反转)、AOP(面向切面编程)
			1.Spring (1)Spring是什么? 是一个轻量级的.用来简化企业级应用开发的开发框架. 注: a.简化开发: Spring对常用的api做了简化,比如,使用Spring jdbc来访问数据库 ... 
- Material适配2 - 高级篇
			版权声明: 欢迎转载,但请保留文章原始出处 作者:GavinCT 出处:http://www.cnblogs.com/ct2011/p/4493439.html 继续Material系列,先从Tool ... 
- Eclipse 下载、安装、取消自动更新、设置编码、关联Tomcat
			1.下载 (1)安装jdk 如果进行web开发,下载java se 版本的jdk即可,不需要像安装 java se 一样安装java ee(里面大多是接口和抽象类).关于java ee的依赖问题有两种 ... 
- Linux安装Tomcat服务器发布项目教程
			前面小Alan跟大家聊了在Linux服务器上jdk运行环境的安装以及redis非关系型数据库的安装,今天继续跟大家聊聊Tomcat的安装,以及将我们的项目发布上去并成功的访问. 第一步:将tomcat ... 
- Oracle EBS 系统仅存在英文的环境
			系统管理员 应用服务器 adadmin 编译 
- Python爬虫实战---抓取图书馆借阅信息
			Python爬虫实战---抓取图书馆借阅信息 原创作品,引用请表明出处:Python爬虫实战---抓取图书馆借阅信息 前段时间在图书馆借了很多书,借得多了就容易忘记每本书的应还日期,老是担心自己会违约 ... 
- 【Oracle】存储过程写法小例子
			1.存储过程的基本语法: CREATE OR REPLACE PROCEDURE 存储过程名(param1 in type,param2 out type) IS 变量1 类型(值范围); 变量2 类 ... 
