POJ 2778 DNA Sequence(AC自动机+矩阵)
【题目链接】 http://poj.org/problem?id=2778
【题目大意】
给出一些字符串,求不包含这些字符串的长度为n的字符串的数量
【题解】
我们将所有串插入自动机计算match,对于自动机上所有节点构建转移矩阵,
对于得到的可达矩阵我们求n长路的数量,统计0到各个点的n长路之和就是答案。
【代码】
#include <cstdio>
#include <cstring>
using namespace std;
const int N=110;
typedef long long LL;
LL P=100000LL;
struct mat{
int n;
LL num[110][110];
void init0(int t){
n=t;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
num[i][j]=0;
}
void init1(int t){
n=t;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
if(i!=j)num[i][j]=0;else num[i][j]=1;
}
mat operator = (const struct mat p){
n=p.n;
for(int i=0;i<n;i++)for(int j=0;j<n;j++)num[i][j]=p.num[i][j];
}
mat operator * (const struct mat p)const{
struct mat ans;
ans.init0(n);
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
for(int k=0;k<n;k++)
ans.num[i][j]=(ans.num[i][j]+num[i][k]*p.num[k][j])%P;
return ans;
}
mat operator ^(int t)const{
struct mat ans,now;
ans.init1(n);
now.n=n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
now.num[i][j]=num[i][j];
while(t>0){
if(t&1)ans=ans*now;
now=now*now;
t>>=1;
}return ans;
}
}mat;
namespace AC_DFA{
const int Csize=4;
int tot,son[N][Csize],sum[N],fail[N],q[N],ans[N],match[N];
void Initialize(){
memset(sum,0,sizeof(int)*(tot+1));
memset(ans,0,sizeof(int)*(tot+1));
memset(match,0,sizeof(int)*(tot+1));
memset(fail,0,sizeof(int)*(tot+1));
for(int i=0;i<=tot;i++)for(int j=0;j<Csize;j++)son[i][j]=0;
tot=0; fail[0]=-1;
}
inline int Tr(char ch){
if(ch=='A')return 0;
if(ch=='T')return 1;
if(ch=='C')return 2;
if(ch=='G')return 3;
}
int Insert(char *s){
int x=0;
for(int l=strlen(s),i=0,w;i<l;i++){
if(!son[x][w=Tr(s[i])]){
son[x][w]=++tot;
}x=son[x][w];
}sum[x]++;
return x;
}
void MakeFail(){
int h=1,t=0,i,j,x=0;
for(i=0;i<Csize;i++)if(son[0][i]){
q[++t]=son[0][i];
match[son[0][i]]=sum[son[0][i]]?son[0][i]:match[fail[son[0][i]]];
}
while(h<=t)for(x=q[h++],i=0;i<Csize;i++)
if(son[x][i]){
fail[son[x][i]]=son[fail[x]][i],q[++t]=son[x][i];
match[son[x][i]]=sum[son[x][i]]?son[x][i]:match[fail[son[x][i]]];
}else son[x][i]=son[fail[x]][i];
}
}
using namespace AC_DFA;
char s[20];
void BuildMat(){
mat.init0(tot+1);
for(int i=0;i<=tot;i++){
if(match[i])continue;
for(int j=0;j<Csize;j++){
if(!match[son[i][j]])mat.num[i][son[i][j]]++;
}
}
}
int n,m;
int main(){
while(~scanf("%d%d",&m,&n)){
Initialize();
while(m--){scanf("%s",s);Insert(s);}
MakeFail(); BuildMat();
mat=mat^n; int ans=0;
for(int i=0;i<mat.n;i++)ans=(ans+mat.num[0][i])%P;
printf("%d\n",ans);
}return 0;
}
POJ 2778 DNA Sequence(AC自动机+矩阵)的更多相关文章
- poj 2778 DNA Sequence ac自动机+矩阵快速幂
链接:http://poj.org/problem?id=2778 题意:给定不超过10串,每串长度不超过10的灾难基因:问在之后给定的长度不超过2e9的基因长度中不包含灾难基因的基因有多少中? DN ...
- POJ 2778 DNA Sequence (AC自动机,矩阵乘法)
题意:给定n个不能出现的模式串,给定一个长度m,要求长度为m的合法串有多少种. 思路:用AC自动机,利用AC自动机上的节点做矩阵乘法. #include<iostream> #includ ...
- poj 2778 DNA Sequence AC自动机DP 矩阵优化
DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 11860 Accepted: 4527 Des ...
- POJ 2778 DNA Sequence ( AC自动机、Trie图、矩阵快速幂、DP )
题意 : 给出一些病毒串,问你由ATGC构成的长度为 n 且不包含这些病毒串的个数有多少个 分析 : 这题搞了我真特么久啊,首先你需要知道的前置技能包括 AC自动机.构建Trie图.矩阵快速幂,其中矩 ...
- poj 2778 DNA Sequence AC自动机
DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 11860 Accepted: 4527 Des ...
- POJ 2778 DNA Sequence (AC自动机+DP+矩阵)
题意:给定一些串,然后让你构造出一个长度为 m 的串,并且不包含以上串,问你有多少个. 析:很明显,如果 m 小的话 ,直接可以用DP来解决,但是 m 太大了,我们可以认为是在AC自动机图中,根据离散 ...
- POJ2278 DNA Sequence —— AC自动机 + 矩阵优化
题目链接:https://vjudge.net/problem/POJ-2778 DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Tota ...
- POJ 2778 DNA Sequence (AC自己主动机 + dp)
DNA Sequence 题意:DNA的序列由ACTG四个字母组成,如今给定m个不可行的序列.问随机构成的长度为n的序列中.有多少种序列是可行的(仅仅要包括一个不可行序列便不可行).个数非常大.对10 ...
- [poj2778]DNA Sequence(AC自动机+矩阵快速幂)
题意:有m种DNA序列是有疾病的,问有多少种长度为n的DNA序列不包含任何一种有疾病的DNA序列.(仅含A,T,C,G四个字符) 解题关键:AC自动机,实际上就是一个状态转移图,注意能少取模就少取模, ...
- POJ - 2778 ~ HDU - 2243 AC自动机+矩阵快速幂
这两题属于AC自动机的第二种套路通过矩阵快速幂求方案数. 题意:给m个病毒字符串,问长度为n的DNA片段有多少种没有包含病毒串的. 根据AC自动机的tire图,我们可以获得一个可达矩阵. 关于这题的t ...
随机推荐
- js_模块化
https://www.cnblogs.com/scq000/p/10647128.html
- Java并发编程(3) JUC中的锁
一 前言 前面已经说到JUC中的锁主要是基于AQS实现,而AQS(AQS的内部结构 .AQS的设计与实现)在前面已经简单介绍过了.今天记录下JUC包下的锁是怎么基于AQS上实现的 二 同步锁 同步锁不 ...
- Memcache 内存分配策略和性能(使用)状态检查【转】
前言: 一直在使用Memcache,但是对其内部的问题,如它内存是怎么样被使用的,使用一段时间后想看看一些状态怎么样?一直都不清楚,查了又忘记,现在整理出该篇文章,方便自己查阅.本文不涉及安装.操作. ...
- 【日记】NOIP2018
day-2: 最后一次走出机房,刚下过几天的雨,感受到的是彻骨的寒意.下午离开教室,跟班主任请了接下来几天的假,班主任斜视了我一眼,哼了一声,确认了一下,不再理会我了.班里的同学或是忙着自己的作业,或 ...
- 形参前的&&啥意思?
C++2011标准的 右值引用 语法 去搜索“c++11右值引用” 右值引用,当传入临时对象时可以避免一次拷贝. 右值引用.举个例子 C/C++ code ? 1 2 3 4 5 6 7 8 // ...
- python中set
集合update方法:是把要传入的元素拆分,做为个体传入到集合中,例如: >>> a = set('boy') >>> a.update('python') > ...
- Linux网络综合命令——IP
1.作用 ip是iproute2软件包里面的一个强大的网络配置工具,它能够替代一些传统的网络管理工具,例如ifconfig.route等,使用权限为超级用户.几乎所有的Linux发行版本都支持该命令. ...
- php 的swoole 和websocket 连接wss
1. 下载证书 $serv = new swoole_server('0.0.0.0', 9501, SWOOLE_PROCESS, SWOOLE_SOCK_TCP | SWOOLE_SSL); $s ...
- SQLSERVER 2008 编辑所有或者任意行
选中表 右键选择“编辑前200行”,然后选择左上角的 sql图标,然后在右侧的SQL语句去掉 top 200 然后执行查询 就可以编辑所有的行了,可以选择自己需要写SQL,然后查询编辑. 第二种方法: ...
- python 统计MySQL表信息
一.场景描述 线上有一台MySQL服务器,里面有几十个数据库,每个库有N多表. 现在需要将每个表的信息,统计到excel中,格式如下: 库名 表名 表说明 建表语句 db1 users 用户表 CRE ...