poj2778(AC 自动机)
poj2778
题意
构造只包含 \(A, T, C, G\) 的字符串,且满足不出现指定的一些字符串,问长度为 \(n\) 的字符串有多少种 ?
分析
AC 自动机 + 矩阵快速幂的神题 ,知识点很多。。。
AC 自动机为了给不同的状态之间建边,矩阵快速幂是为了加速状态转移。
比如说一共有 \(5\) 个状态,我要从 状态 \(0\) 转移到 状态 \(4\) ,从 \(0\) 出发,可以先转移到 \(0\) 再转移到 \(4\) ,也可以先转移到 \(1\) 再转移到 \(4\) ,后面类似。
建一个邻接矩阵,\(mat[i][j]\) 表示 \(i\) 转移到 \(j\) 的方案数,想象一下矩阵相乘的情况,\(mat[0][4]\) 的计算过程,神奇。。。
code
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<queue>
using namespace std;
typedef long long ll;
const int MAXN = 105;
const int MOD = 1e5;
int n, m;
struct Matrix {
ll mat[MAXN][MAXN];
void init() {
memset(mat, 0, sizeof mat);
}
};
Matrix operator*(Matrix A, Matrix B) {
Matrix C;
C.init();
for(int i = 0; i < n; i++) {
for(int j = 0; j < n; j++) {
for(int k = 0; k < n; k++) {
C.mat[i][j] = (C.mat[i][j] + A.mat[i][k] * B.mat[k][j]) % MOD;
}
}
}
return C;
}
Matrix operator^(Matrix A, int x) {
Matrix B;
B.init();
for(int i = 0; i < n; i++) B.mat[i][i] = 1;
while(x) {
if(x & 1) B = B * A;
A = A * A;
x >>= 1;
}
return B;
}
struct Trie {
int id[100];
int root, L, nxt[MAXN][4], val[MAXN], fail[MAXN];
int newnode() {
for(int i = 0; i < 4; i++) {
nxt[L][i] = -1;
}
return L++;
}
void init() {
id['A'] = 0; id['T'] = 1; id['C'] = 2; id['G'] = 3;
L = 0;
root = newnode();
memset(val, 0, sizeof val);
}
void insert(char s[15]) {
int len = strlen(s);
int now = root;
for(int i = 0; i < len; i++) {
int d = id[s[i]];
if(nxt[now][d] == -1) nxt[now][d] = newnode();
now = nxt[now][d];
}
val[now] = 1;
}
void build() {
queue<int> Q;
for(int i = 0; i < 4; i++) {
if(nxt[root][i] == -1) nxt[root][i] = root;
else {
fail[nxt[root][i]] = root;
Q.push(nxt[root][i]);
}
}
while(!Q.empty()) {
int now = Q.front(); Q.pop();
if(val[fail[now]]) val[now] = 1;
for(int i = 0; i < 4; i++) {
if(nxt[now][i] == -1) nxt[now][i] = nxt[fail[now]][i];
else {
fail[nxt[now][i]] = nxt[fail[now]][i];
Q.push(nxt[now][i]);
}
}
}
}
Matrix buildMatrix() {
Matrix A; A.init();
for(int i = 0; i < L; i++) {
for(int j = 0; j < 4; j++) {
if(!val[i] && !val[nxt[i][j]]) {
A.mat[i][nxt[i][j]]++;
}
}
}
return A;
}
}trie;
int main() {
trie.init();
int k;
scanf("%d%d", &m, &k);
for(int i = 0; i < m; i++) {
char s[15];
scanf("%s", s);
trie.insert(s);
}
trie.build();
Matrix A = trie.buildMatrix();
n = trie.L;
A = A ^ k;
int ans = 0;
for(int i = 0; i < n; i++) {
ans = (ans + A.mat[0][i]) % MOD;
}
printf("%d\n", ans);
return 0;
}
poj2778(AC 自动机)的更多相关文章
- poj2778 ac自动机+矩阵快速幂
给m个子串,求长度为n的不包含子串的母串数,最直接的应该是暴搜,肯定tle,考虑用ac自动机 将子串建成字典树,通过next表来构造矩阵,然后用矩阵快速幂求长度为n的数量 邻接矩阵https://we ...
- poj2778 AC自动机
以下内容均为转载,,只有代码是自己写的=-= http://blog.csdn.net/morgan_xww/article/details/7834801 转载地址 博主写的很好 ------- ...
- 【POJ2778】DNA Sequence(AC自动机,DP)
题意: 生物课上我们学到,DNA序列中只有A, C, T和G四种片段. 经科学发现,DNA序列中,包含某些片段会产生不好的基因,如片段"ATC"是不好片段,则"AGATC ...
- 【POJ2778】AC自动机+矩阵乘法
DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14758 Accepted: 5716 Descrip ...
- POJ2778 DNA Sequence(AC自动机 矩阵)
先使用AC自动机求得状态转移关系,再建立矩阵,mat[i][j]表示一步可从i到j且i,j节点均非终止字符的方案数,则此矩阵的n次方表示n步从i,到j的方法数. #include<cstdio& ...
- POJ2778 DNA Sequence(AC自动机+矩阵快速幂)
题目给m个病毒串,问不包含病毒串的长度n的DNA片段有几个. 感觉这题好神,看了好久的题解. 所有病毒串构造一个AC自动机,这个AC自动机可以看作一张有向图,图上的每个顶点就是Trie树上的结点,每个 ...
- poj2778(AC自动机+矩阵快速幂)
题意:给你n个字符串,问你长度为m的字符串且字符串中不含有那n个子串的字符串的数量 解题思路:这道题一开始就不太懂,还以为是组合数学的题目,后面看了别人的博客,才知道这是属于AC自动机的另一种用法,是 ...
- 【POJ2778】DNA Sequence 【AC自动机,dp,矩阵快速幂】
题意 题目给出m(m<=10)个仅仅由A,T,C,G组成的单词(单词长度不超过10),然后给出一个整数n(n<=2000000000),问你用这四个字母组成一个长度为n的长文本,有多少种组 ...
- poj2778 DNA Sequence【AC自动机】【矩阵快速幂】
DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 19991 Accepted: 7603 Des ...
随机推荐
- hbase(0.94) get、scan源码分析
简介 本文是需要用到hbase timestamp性质时研究源码所写.内容有一定侧重.且个人理解不算深入,如有错误请不吝指出. 如何看源码 hbase依赖很重,没有独立的client包.所以目前如果在 ...
- Netscaler重置密码的方法
Netscaler重置密码的方法 http://blog.51cto.com/caojin/1898401 有时候我们会碰到忘记Netscaler的密码,或接手别人的设备而不知道密码的情况.在这种情况 ...
- C&C++——C函数与C++函数相互调用问题
C C++相互调用 在项目中融合C和C++有时是不可避免的,在调用对方的功能函数的时候,或许会出现这样那样的问题,但只要我的C代码和我的C++代码分别都能成功编译,那其他就不是问题.近来在主程序是C语 ...
- Java代码管理工具SVN系列
SVN是Subversion的简称,是一个开放源代码的版本控制系统,相较于RCS.CVS,它采用了分支管理系统,它的设计目标就是取代CVS.互联网上很多版本控制服务已从CVS迁移到Subversion ...
- Elasticsearch报错
[2018-07-12T10:32:47,642][INFO ][o.e.b.BootstrapChecks ] [VfCcJIq] bound or publishing to a non-loop ...
- [模拟赛] GotoAndPlay
GotoAndPlay 10月3日,在杭州市西湖景区,一只小松鼠不停地接受一道道食物,花生. 玉米.饼干,可谓来者不拒,憨态可掬的模样吸引了众多围观者... Description 小松鼠终于吃撑了, ...
- [poj 2104]主席树+静态区间第k大
题目链接:http://poj.org/problem?id=2104 主席树入门题目,主席树其实就是可持久化权值线段树,rt[i]维护了前i个数中第i大(小)的数出现次数的信息,通过查询两棵树的差即 ...
- 问题总结——window平台下grunt\bower安装后无法运行的问题
一.问题: 安装grunt或者bower后,在cmd控制台运行grunt -version 或者 bower -v会出现:“xxx不是内部或外部命令,也不是可运行的程序或批处理文件”,
- css中文本超出部分省略号代替
p{ width: 100px; //设置p标签宽度 white-space: nowrap; //文本超出P标签宽度不换行,而是溢出 overflow: hidden; //文本超出P标签,超出部分 ...
- spring+jersey+c3p0构建restful webservice(数据源采用c3p0)
项目下载地址: http://files.cnblogs.com/files/walk-the-Line/payment.zip