poj 1625 (AC自动机好模版,大数好模版)
给n个字母,构成长度为m的串,总共有n^m种。给p个字符串,问n^m种字符串中不包含(不是子串)这p个字符串的个数。
将p个不能包含的字符串建立AC自动机,每个结点用val值来标记以当前节点为后缀的字符串是否包含非法字符串(p个字符串中的任何一个)。
状态转移方程:f(i, j) += f(i-1, k)
f(i, j)表示长度为i的字符串,结尾为字符j,方程j和k的关系可以从自动机中失配关系直接获得(j是k的后继结点)。
总之感觉是好东西,快存下来
大数模版:
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std; struct BigInteger{
int A[];
enum{MOD = };
BigInteger(){memset(A, , sizeof(A)); A[]=;}
void set(int x){memset(A, , sizeof(A)); A[]=; A[]=x;}
void print(){
printf("%d", A[A[]]);
for (int i=A[]-; i>; i--){
if (A[i]==){printf(""); continue;}
for (int k=; k*A[i]<MOD; k*=) printf("");
printf("%d", A[i]);
}
printf("\n");
}
int& operator [] (int p) {return A[p];}
const int& operator [] (int p) const {return A[p];}
BigInteger operator + (const BigInteger& B){
BigInteger C;
C[]=max(A[], B[]);
for (int i=; i<=C[]; i++)
C[i]+=A[i]+B[i], C[i+]+=C[i]/MOD, C[i]%=MOD;
if (C[C[]+] > ) C[]++;
return C;
}
BigInteger operator * (const BigInteger& B){
BigInteger C;
C[]=A[]+B[];
for (int i=; i<=A[]; i++)
for (int j=; j<=B[]; j++){
C[i+j-]+=A[i]*B[j], C[i+j]+=C[i+j-]/MOD, C[i+j-]%=MOD;
}
if (C[C[]] == ) C[]--;
return C;
}
};
int main() {
BigInteger a, b;
a.set(); b.set();
(a+b).print(); return ;
}
本题答案(内含AC自动机模版):
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
typedef unsigned char uchar; struct AC_Automata {
#define N 102
int ch[N][], val[N], last[N], f[N], sz;
void clear() { sz = ; memset(ch[], , sizeof(ch[])); } int hash[], M;
void set_hash(int n, uchar s[]) {
M = n; for (int i=; i<n; i++) hash[s[i]] = i;
}
void insert(uchar s[], int v) {
int u = ;
for (int i=; s[i]; i++) {
int c = hash[s[i]];
if (!ch[u][c]) {
memset(ch[sz], , sizeof(ch[sz]));
val[sz] = ;
ch[u][c] = sz++;
}
u = ch[u][c];
}
val[u] = v; //标记当前串为非法的
}
void build() {
queue<int> q;
f[] = ;
for (int c=; c<M; c++) {
int u = ch[][c];
if (u) { f[u] = last[u] = ; q.push(u); }
}
while (!q.empty()) {
int r = q.front(); q.pop();
for (int c=; c<M; c++) {
int u = ch[r][c];
val[r] = val[r] || val[f[r]]; //判断当前结点是否有非法后缀
if (!u) {
ch[r][c] = ch[f[r]][c];
continue;
}
q.push(u);
f[u] = ch[f[r]][c];
last[u] = val[f[u]] ? f[u] : last[f[u]];
}
}
}
} ac;
struct BigInteger{
int A[];
enum{MOD = };
BigInteger(){memset(A, , sizeof(A)); A[]=;}
void set(int x){memset(A, , sizeof(A)); A[]=; A[]=x;}
void print(){
printf("%d", A[A[]]);
for (int i=A[]-; i>; i--){
if (A[i]==){printf(""); continue;}
for (int k=; k*A[i]<MOD; k*=) printf("");
printf("%d", A[i]);
}
printf("\n");
}
int& operator [] (int p) {return A[p];}
const int& operator [] (int p) const {return A[p];}
BigInteger operator + (const BigInteger& B){
BigInteger C;
C[]=max(A[], B[]);
for (int i=; i<=C[]; i++)
C[i]+=A[i]+B[i], C[i+]+=C[i]/MOD, C[i]%=MOD;
if (C[C[]+] > ) C[]++;
return C;
}
BigInteger operator * (const BigInteger& B){
BigInteger C;
C[]=A[]+B[];
for (int i=; i<=A[]; i++)
for (int j=; j<=B[]; j++){
C[i+j-]+=A[i]*B[j], C[i+j]+=C[i+j-]/MOD, C[i+j-]%=MOD;
}
if (C[C[]] == ) C[]--;
return C;
}
};
int n, m, p;
uchar s[]; int main() { while (scanf("%d %d %d ", &n, &m, &p) == ) {
ac.clear();
cin >> s; ac.set_hash(n, s);
while (p--) {
cin >> s; ac.insert(s, );
}
ac.build(); BigInteger f[][];
f[][].set(); for (int i=; i<=m; i++)
for (int j=; j<ac.sz; j++)
for (int k=; k<n; k++) {
int u = ac.ch[j][k];
if (!ac.val[u]) f[i][u] = f[i][u] + f[i-][j];
}
BigInteger ans;
for (int i=; i<ac.sz; i++)
if (!ac.val[i]) ans = ans + f[m][i];
ans.print();
}
return ;
}
poj 1625 (AC自动机好模版,大数好模版)的更多相关文章
- Censored! POJ - 1625 AC自动机+大数DP
题意: 给出一n种字符的字典,有p个禁用的单词, 问能组成多少个不同的长度为m的合法字符串.(m<=50) 题解: 是不是个我们之前做的题目非常非常像,题意都一样. 直接将上次写的AC自动机+矩 ...
- Censored! - POJ 1625(ac自动机+简单dp+高精度运算)
题目大意:首先给一个字符集合,这个集合有N个字符,然后需要一个长度为M的句子,但是据子里面不能包含的串有P个,每个串里面的字符都是有字符集和里面的字符构成的,现在想知道最多能构造多少个不重复的句子. ...
- POJ 3691 (AC自动机+状态压缩DP)
题目链接: http://poj.org/problem?id=3691 题目大意:给定N个致病DNA片段以及一个最终DNA片段.问最终DNA片段最少修改多少个字符,使得不包含任一致病DNA. 解题 ...
- POJ 2778 (AC自动机+矩阵乘法)
POJ 2778 DNA Sequence Problem : 给m个只含有(A,G,C,T)的模式串(m <= 10, len <=10), 询问所有长度为n的只含有(A,G,C,T)的 ...
- POJ 2896 AC自动机 or 暴力
DESCRIPTION :大意是说.给你n个代表病毒的字符串.m个表示网站的字符串.让你计算有多少个网站被病毒感染了.被那些病毒感染了. 刚开始就想暴力.然而,忽略了条件:每个网站最多有三个病毒.于是 ...
- DNA Sequence POJ - 2778 AC自动机 && 矩阵快速幂
It's well known that DNA Sequence is a sequence only contains A, C, T and G, and it's very useful to ...
- poj 4052(ac自动机)
题意:自己百度吧!! 分析:就是通过它的fail指针来找出它的子串就行了,这题其实不难的.这好像还是金华邀请赛的题哦! 代码实现: #include<cstdio> #include< ...
- poj 2778 AC自动机+矩阵快速幂
题目链接:https://vjudge.net/problem/POJ-2778 题意:输入n和m表示n个病毒,和一个长为m的字符串,里面只可以有'A','C','G','T' 这四个字符,现在问这个 ...
- DNA Sequence POJ - 2778 AC 自动机 矩阵乘法
定义重载运算的时候一定要将矩阵初始化,因为这个调了一上午...... Code: #include<cstdio> #include<algorithm> #include&l ...
随机推荐
- Python装饰器笔记
DRY(Don't Repeat Yourself)原则: 一般是指在写代码的时候尽量避免重复的实现.违反DRY原则导致的坏处很容易理解,例如维护困难,修改时一旦遗漏就会产生不易察觉的问题. 一.函数 ...
- 8.Android之日期DatePicker和时间TimeTicker控件学习
手机设置时间日期很普遍,今天就梳理下. 首先在拖入一个按钮 ,日期和时间控件到工程里,如图: 代码如下: <?xml version="1.0" encoding=" ...
- 学习笔记-动态树Link-Cut-Tree
--少年你有梦想吗? --少年你听说过安利吗? 安利一个集训队讲解:http://wenku.baidu.com/view/75906f160b4e767f5acfcedb 关于动态树问题,有多种方法 ...
- javaIO(三)
- 慎用 Enum.GetHashCode()
公司里遗留下了相当多的 Enum.GetHashCode()来获取枚举值的代码 但是这会产生装箱行为的!!因为Enum是值类型,GetHashCode()是Object的方法,调用GetHashCod ...
- Android 实现卫星菜单
步骤:一:自定义ViewGroup 1.自定义属性 a.attr.xml b.在布局文件中使用activity_main.xml c.在自定义控件中进行读取 2.onMeasure 3.onLayou ...
- Matplotlib中文设置
1.中文设置方法,代码前加入语句 from pylab import mpl mpl.rcParams['font.sans-serif'] = ['SimHei'] 2.例子 # -*- codin ...
- mybatis 使用动态SQL
RoleMapper.java public interface RoleMapper { public void add(Role role); public void update(Role ro ...
- mingw32-g++.exe: *: No such file or directory错误解决方法
初次使用CodeBlocks,好不容易把环境配好, 编译没有错误了,但是程序并不生成exe,提示以下问题: mingw32-g++.exe: /W3: No such file or director ...
- 各种工具使用手册:http://www.itshouce.com.cn/linux/linux-tcpdump.html 关于tcpdump!!!!
各种工具使用手册:http://www.itshouce.com.cn/linux/linux-tcpdump.html 关于tcpdump!!!! 实用tcpdump命令 //查看本机与mysql的 ...