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 ...
随机推荐
- ORACLE数据库删除表中记录报record is locked by another user
在操作ORACLE数据库的时候,由于执行完,没有COMMIT,直接把PL/SQL关闭掉,后来导致那张表被锁住,当编辑时就会出现这个信息,record is locked by another user ...
- Dinic问题
问题:As more and more computers are equipped with dual core CPU, SetagLilb, the Chief Technology Offic ...
- 【poj1740】 A New Stone Game
http://poj.org/problem?id=1740 (题目链接) 男人八题之一 题意 对于n堆石子,每堆若干个,两人轮流操作,每次操作分两步,第一步从某堆中去掉至少一个,第二步(可省略)把该 ...
- 使用python来调试串口
串口模块的波特率比较特别,找了几个串口工具都不支持...所以,干脆用python自己来写了,其实已经好奇好久了,别人的工具各种不顺手. 需要pyserial的支持,兼容各种平台,不需要新编译二进制文件 ...
- UVA 1149 Bin Packing
传送门 A set of n 1-dimensional items have to be packed in identical bins. All bins have exactly the sa ...
- php5.5安装及phpmyadmin&nginx配置php模块
安装php5.5: 下载源地址:rpm -Uvh rpm包安装:yum install php55w.x86_64 php55w-cli.x86_64 php55w-common.x86_64 php ...
- React Native 开发之 (02) 用Sublime 3作为React Native的开发IDE
Sublime Text是一个代码编辑器.也是HTML和散文先进的文本编辑器.漂亮的用户界面和非凡的功能,例如:迷你地图,多选择Python插件,代码段等等.完全可自定义键绑定,菜单和工具栏等等.漂亮 ...
- 轻量级应用开发之(02)UIView
一 控件 1.屏幕上的所有UI元素都叫做控件(也有叫做视图.组件)比如按钮(UIButton).文本(UILabel)都是控件. 2.控件的共同属性有哪些? 尺寸,位置,背景色 3. 苹果将控件的共同 ...
- 微型 ORM-FluentData 温故知新系列
http://www.cnblogs.com/_popc/archive/2012/12/26/2834726.html 引言:FluentData 是微型 ORM(micro-ORM)家族的一名新成 ...
- jquery------.resizable()的使用
index.jsp //加上这两行代码,右下角会有样式效果<link rel="stylesheet" href="//code.jquery.com/ui/1.1 ...