URAL 1158 AC自动机上的简单DP+大数
题目大意
在一种语言中的字母表中有N(N<=50)个字母,每个单词都由M(M<=50)个字母构成,因此,一共可以形成N^M个单词。但是有P(P<=10)个串是被禁止的,也就是说,任何单词的子串都不能包含这P个串中的任意一个。问按照上述规则,能产生的合法串一共有多少个? 例如:N=3 M=3 P=3 字母表中的三个字符是QWE 被禁止的串为”QQ”,”WEE”,”Q”,则合法的串一共有7个。
这题目相当于通过步数对AC自动机上每一个点的状态进行DP
dp[i][j]表示到达i这个点,走了j步存在多少种方法
总是从上一步推到下一步
写成滚动数组也不会有问题
这里要注意在AC自动机更新操作时,要将每个串末尾能到达的fail位置也记上标记,题解中有注释解释
#include <cstdio>
#include <cstring>
#include <queue>
#include <vector>
#include <iostream>
#include <algorithm>
#include <map>
using namespace std;
#define clr(x) memset(x , 0 , sizeof(x))
#define set(x) memset(x , -1 , sizeof(x))
typedef long long LL ;
#define rep( i , a , b ) for ( int i = a ; i < b ; ++ i )
#define For( i , a , b ) for ( int i = a ; i <= b ; ++ i )
#define rev( i , a , b ) for ( int i = a ; i >= b ; -- i )
map<char , int> mp; const int CHAR_SIZE = ;
const int MAX_SIZE = ;
const int M = ;
int n,m,p; struct BigInt {
int wei , a[];
BigInt(){
wei = ;
memset(a , , sizeof(a));
}
void init(){
wei = ;
memset(a , , sizeof(a));
}
void print(){
for(int i=wei- ; i>= ; i--) printf("%d" , a[i]);
printf("\n");
}
BigInt operator+(BigInt m){
BigInt ret;
int ma = max(wei , m.wei);
for(int i= ; i<ma ; i++){
ret.a[i] += a[i]+m.a[i];
if(ret.a[i]>=) ret.a[i+]++,ret.a[i]-=;
}
if(ret.a[ma]) ret.wei = ma+;
else ret.wei = ma;
return ret;
}
}; struct AC_Machine{
int ch[MAX_SIZE][CHAR_SIZE] , val[MAX_SIZE] , fail[MAX_SIZE];
int sz; void init(){
sz = ;
clr(ch[]) , clr(val);
} void insert(char *s){
int n = strlen(s);
int u= ;
for(int i= ; i<n ; i++){
int c = mp[s[i]];
if(!ch[u][c]){
clr(ch[sz]);
val[sz] = ;
ch[u][c] = sz++;
}
u = ch[u][c];
}
val[u] = ;
} void get_fail(){
queue<int> Q;
fail[] = ;
for(int c= ; c<n ; c++){
int u = ch[][c];
if(u){Q.push(u);fail[u]=;}
}
while(!Q.empty()){
int r = Q.front();
Q.pop();
//比如is, history都是非法的,history中有is,那么history访问到它的s时也是非法的
val[r] |= val[fail[r]];
for(int c= ; c<n ; c++){
int u = ch[r][c];
if(!u){ch[r][c] = ch[fail[r]][c]; continue;}
fail[u] = ch[fail[r]][c];
Q.push(u);
}
}
}
}ac; char str[];
BigInt dp[MAX_SIZE][]; BigInt solve(int n , int step)
{
for(int i= ; i<ac.sz ; i++)
for(int j= ; j<=step ; j++){
dp[i][j].init();
}
dp[][].wei = dp[][].a[] = ;
for(int i= ; i<=step ; i++){
for(int j= ; j<ac.sz ; j++){
for(int k= ; k<n ; k++)
if(!ac.val[ac.ch[j][k]]){
dp[ac.ch[j][k]][i] = dp[ac.ch[j][k]][i]+dp[j][i-];
}
}
}
BigInt ret = BigInt();
for(int j= ; j<ac.sz ; j++) ret = ret+dp[j][step];
return ret;
} int main()
{
// freopen("in.txt" , "r" , stdin);
// freopen("out.txt" , "w" , stdout);
while(~scanf("%d%d%d" , &n , &m , &p)){
mp.clear();
getchar();
gets(str);
for(int i= ; i<strlen(str) ; i++) mp[str[i]] = i;
ac.init();
for(int i= ; i<=p ; i++){
gets(str);
ac.insert(str);
}
ac.get_fail();
BigInt ret = solve(n , m);
ret.print();
}
return ;
}
URAL 1158 AC自动机上的简单DP+大数的更多相关文章
- 【洛谷4045】[JSOI2009] 密码(状压+AC自动机上DP)
点此看题面 大致题意: 给你\(n\)个字符串,问你有多少个长度为\(L\)的字符串,使得这些字符串都是它的子串.若个数不大于\(42\),按字典序输出所有方案. 状压 显然,由于\(n\)很小,我们 ...
- POJ 3691 AC自动机上的dp
题目大意: 给定一些不合理的DNA序列,再给一段较长的dna序列,问最少修改几次可以使序列中不存在任何不合理序列,不能找到修改方法输出-1 这里你修改某一个点的DNA可能会影响后面,我们不能单纯的找匹 ...
- HNU 13108-Just Another Knapsack Problem (ac自动机上的dp)
题意: 给你一个母串,多个模式串及其价值,求用模式串拼接成母串(不重叠不遗漏),能获得的最大价值. 分析: ac自动机中,在字典树上查找时,用dp,dp[i]拼成母串以i为结尾的子串,获得的最大价值, ...
- 【BZOJ1030】[JSOI2007] 文本生成器(AC自动机上跑DP)
点此看题面 大致题意: 给你\(N\)个字符串(只含大写字母),要你求出有多少个由\(M\)个大写字母构成的字符串含有这\(N\)个字符串中的至少一个. \(AC\)自动机 看到题目,应该比较容易想到 ...
- Passwords Gym - 101174E (AC自动机上DP)
Problem E: Passwords \[ Time Limit: 1 s \quad Memory Limit: 256 MiB \] 题意 给出两个正整数\(A,B\),再给出\(n\)个字符 ...
- BZOJ 1030 文本生成器 | 在AC自动机上跑DP
题目: http://www.lydsy.com/JudgeOnline/problem.php?id=1030 题解: 鸽 #include<cstdio> #include<al ...
- bzoj [Sdoi2014]数数 AC自动机上dp
[Sdoi2014]数数 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1264 Solved: 636[Submit][Status][Discu ...
- 【Luogu】P3311数数(AC自动机上DP)
题目链接 蒟蒻今天终于学会了AC自动机,感觉很稳 (后一句愚人节快乐) 这题开一个f[i][j][k]表示有没有受到限制,正在枚举第j位,来到了AC自动机的第k个节点 的方案数 随后可以刷表更新 注意 ...
- hdu 3992 AC自动机上的高斯消元求期望
Crazy Typewriter Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
随机推荐
- Android 视频投射之NanoHTTPD
Android 视频投射之NanoHTTPD 号称用一个java文件实现Http服务器 有必要对其源码及例子进行分析 public abstract class NanoHTTPD { //异步执行请 ...
- SecureCRT设置
SecureCRT设置 文章来源:http://blog.csdn.net/dongqinliuzi/article/details/39890569 本文主要介绍SecureCRT的使用方法和技巧. ...
- hiho_1086_browser_caching
题目 浏览器有一个cache,可以存放M(M <= 5000) 个url地址(url为长度小于30的字符串).现在进行N(N <= 20000)次访问,每一个访问,如果访问的url在cac ...
- iOS开发 判断用户是否开启了定位
- (BOOL)achiveUserLocationStart { CLAuthorizationStatus status = [CLLocationManager authorizationSta ...
- el 和 fmt 常用
EL表达式入门 转自 http://blog.chinaunix.net/uid-9789791-id-1997374.html 隐含对象: pageContext: pageContext对象 pa ...
- git服务器搭建-new
http://blog.chinaunix.net/uid-26729093-id-4652536.html 基本命令使用:http://www.cnblogs.com/xdao/p/linux_gi ...
- dedecms 模板文件不存在,无法解析文档"的终极各种解决办法
方法一:[此对应喜欢把模板文件使用".html"的格式,] /include/arc.archives.class.php 556行 if (!preg_match(&qu ...
- 创建和使用Windows静态链接库
首先明确这篇文章的目的,我希望大家能够通过这篇文章了解一下如何在实际工作中创建和使用Windows平台下的静态链接库.关于链接库的概念,希望大家参考维基百科"Library"词条( ...
- RESEACH PAPER
个,proquest的username和password赫然在目,别急,再看第4个结 果"HB Thompson Subscription Online Databases", ...
- input上传图片预览
<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <t ...