后缀自动机&序列自动机综合
好像序列自动机还没有写过…
串长为n的串共有n+1个节点,除了串中的n个节点,还有一个空的根节点放在串首。每个节点至多有26条出边,每条边连向它之后的第一个字符。
串中的任意一个子序列对应了一条根到某个节点的路径。且每条路径对应一个不同的子序列。
每个节点的parent是这个字母上一次出现的位置。更新只要沿parent指针扫描即可。
FJOI2016 所有公共子序列问题
这题暴力建trie能过80真是悲伤(因为按FJOI命题风格这题没有写数据范围
建完序列自动机暴力DP即可
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <vector>
#include <limits>
#include <set>
#include <map>
using namespace std;
#define SZ 3333
int ys[2333],fys[2333];
#define M 60
struct Seq_A_M
{
int par[SZ],ch[SZ][62],lst[62],C,rot;
Seq_A_M()
{
C=rot=1;
for(int i=0;i<M;i++) lst[i]=rot;
}
void ins(char c)
{
++C; par[C]=lst[c];
for(int i=0;i<M;i++)
{
for(int g=lst[i];g&&!ch[g][c];g=par[g]) ch[g][c]=C;
}
lst[c]=C;
}
}S_A,S_B;
int n,m;
typedef long long ll;
ll dp[SZ][SZ];
char x[SZ],y[SZ];
ll getdp(int a,int b)
{
if(!a||!b) return 0;
if(dp[a][b]>=0) return dp[a][b];
long long ans=1;
for(int i=0;i<M;i++) ans+=getdp(S_A.ch[a][i],S_B.ch[b][i]);
return dp[a][b]=ans;
}
char s[233333];
void tryy(int a,int b,int cl)
{
if(!a||!b) return;
s[cl]=0; printf("%s\n",s);
for(int i=0;i<M;i++)
{
s[cl]=fys[i]; tryy(S_A.ch[a][i],S_B.ch[b][i],cl+1);
}
}
int main()
{
for(int i='A';i<='Z';i++) ys[i]=i-'A', fys[i-'A']=i;
for(int i='a';i<='z';i++) ys[i]=i-'a'+26, fys[i-'a'+26]=i;
int k=0;
scanf("%d%d%s%s%d",&n,&m,x,y,&k);
for(int i=0;i<n;i++) S_A.ins(ys[x[i]]);
for(int i=0;i<m;i++) S_B.ins(ys[y[i]]);
if(k==1) tryy(1,1,0);
memset(dp,-1,sizeof(dp));
printf("%lld\n",getdp(1,1));
}
四校联考 公共串问题
还是暴力DP。注意一些细节。
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <vector>
#include <limits>
#include <set>
#include <map>
using namespace std;
int MOD=998244353;
#define SZ 4008
#define S 26 //字符集
struct AM
{
int rot,ch[SZ][S],C,cnt[SZ];
};
struct SeqAM: public AM
{
int par[SZ],lst[S];
SeqAM()
{
C=rot=1;
for(int i=0;i<S;i++) lst[i]=rot;
}
void ins(char c)
{
++C; par[C]=lst[c];
for(int i=0;i<S;i++)
{
for(int g=lst[i];g&&!ch[g][c];g=par[g]) ch[g][c]=C;
}
lst[c]=C;
}
void getcnt()
{
for(int i=1;i<=C;i++) cnt[i]=1;
for(int i=C;i>=1;i--)
{
for(int j=0;j<S;j++) cnt[i]+=cnt[ch[i][j]], cnt[i]%=MOD;
}
}
}SeqA,SeqB;
struct SufAM: public AM
{
int ml[SZ],fail[SZ],lst,cl,qzh[SZ],od[SZ];
SufAM() {C=lst=rot=1; cl=0;}
void ins(char c)
{
int x=++C,len=++cl,p=lst;
lst=x; ml[x]=len;
for(;p&&!ch[p][c];p=fail[p]) ch[p][c]=x;
if(!p) fail[x]=rot;
else if(ml[ch[p][c]]==ml[p]+1) fail[x]=ch[p][c];
else
{
int chh=ch[p][c],cm=++C;
ml[cm]=ml[p]+1; fail[cm]=fail[chh];
for(int i=0;i<S;i++) ch[cm][i]=ch[chh][i];
fail[chh]=fail[x]=cm;
for(;ch[p][c]==chh;p=fail[p]) ch[p][c]=cm;
}
}
void getcnt()
{
for(int i=0;i<SZ;i++) qzh[i]=0;
for(int i=1;i<=C;i++) qzh[ml[i]]++;
for(int i=1;i<SZ;i++) qzh[i]+=qzh[i-1];
for(int i=1;i<=C;i++) od[qzh[ml[i]]--]=i;
for(int i=1;i<=C;i++) cnt[i]=1;
for(int i=C;i>=1;i--)
{
for(int j=0;j<S;j++) cnt[od[i]]+=cnt[ch[od[i]][j]], cnt[od[i]]%=MOD;
}
}
}SufA,SufB;
void prtat(AM& s)
{
for(int i=1;i<=s.C;i++)
{
for(int j=0;j<S;j++) if(s.ch[i][j]) printf("%d->%d[label=%c];\n",i,s.ch[i][j],j+'a');
}
}
AM *cur,*curb;
int dp2[SZ][SZ];
int dfs(int a,int b)
{
if(!a) return 0;
if(!b) return cur->cnt[a];
if(dp2[a][b]>=0) return dp2[a][b];
int ans=0;
for(int i=0;i<S;i++) ans+=dfs(cur->ch[a][i],curb->ch[b][i]), ans%=MOD;
return dp2[a][b]=ans;
}
int getdp(AM& a,AM& b)
{
cur=&a; curb=&b;
for(int i=1;i<=a.C;i++)
for(int j=1;j<=b.C;j++) dp2[i][j]=-1;
dfs(a.rot,b.rot);
return dp2[a.rot][b.rot];
}
char A[SZ],B[SZ];
void prt(int a)
{
a=(a%MOD+MOD)%MOD;
printf("%d\n",a);
}
int main()
{
scanf("%s%s",A,B);
for(int i=0;A[i];i++) SeqA.ins(A[i]-'a'), SufA.ins(A[i]-'a');
for(int i=0;B[i];i++) SeqB.ins(B[i]-'a'), SufB.ins(B[i]-'a');
SeqA.getcnt(); SeqB.getcnt();
SufA.getcnt(); SufB.getcnt();
int AseqBseq=getdp(SeqA,SeqB);
int AseqBsuf=getdp(SeqA,SufB);
int AsufBseq=getdp(SufA,SeqB);
int AsufBsuf=getdp(SufA,SufB);
prt(AsufBsuf);
prt(AsufBseq);
prt(AseqBsuf);
prt(AseqBseq);
}
bzoj4032 最短不公共子串
#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
#include <string.h>
#include <vector>
#include <limits>
#include <set>
#include <map>
using namespace std;
int MOD=998244353;
#define SZ 4008
#define S 26 //字符集
struct AM
{
int rot,ch[SZ][S],C,cnt[SZ];
};
struct SeqAM: public AM
{
int par[SZ],lst[S];
SeqAM()
{
C=rot=1;
for(int i=0;i<S;i++) lst[i]=rot;
}
void ins(char c)
{
++C; par[C]=lst[c];
for(int i=0;i<S;i++)
{
for(int g=lst[i];g&&!ch[g][c];g=par[g]) ch[g][c]=C;
}
lst[c]=C;
}
void getcnt()
{
for(int i=1;i<=C;i++) cnt[i]=1;
for(int i=C;i>=1;i--)
{
for(int j=0;j<S;j++) cnt[i]+=cnt[ch[i][j]], cnt[i]%=MOD;
}
}
}SeqA,SeqB;
struct SufAM: public AM
{
int ml[SZ],fail[SZ],lst,cl,qzh[SZ],od[SZ];
SufAM() {C=lst=rot=1; cl=0;}
void ins(char c)
{
int x=++C,len=++cl,p=lst;
lst=x; ml[x]=len;
for(;p&&!ch[p][c];p=fail[p]) ch[p][c]=x;
if(!p) fail[x]=rot;
else if(ml[ch[p][c]]==ml[p]+1) fail[x]=ch[p][c];
else
{
int chh=ch[p][c],cm=++C;
ml[cm]=ml[p]+1; fail[cm]=fail[chh];
for(int i=0;i<S;i++) ch[cm][i]=ch[chh][i];
fail[chh]=fail[x]=cm;
for(;ch[p][c]==chh;p=fail[p]) ch[p][c]=cm;
}
}
void getcnt()
{
for(int i=0;i<SZ;i++) qzh[i]=0;
for(int i=1;i<=C;i++) qzh[ml[i]]++;
for(int i=1;i<SZ;i++) qzh[i]+=qzh[i-1];
for(int i=1;i<=C;i++) od[qzh[ml[i]]--]=i;
for(int i=1;i<=C;i++) cnt[i]=1;
for(int i=C;i>=1;i--)
{
for(int j=0;j<S;j++) cnt[od[i]]+=cnt[ch[od[i]][j]], cnt[od[i]]%=MOD;
}
}
}SufA,SufB;
void prtat(AM& s)
{
for(int i=1;i<=s.C;i++)
{
for(int j=0;j<S;j++) if(s.ch[i][j]) printf("%d->%d[label=%c];\n",i,s.ch[i][j],j+'a');
}
}
int dep[SZ][SZ];
int qa[SZ*SZ],qb[SZ*SZ];
int bfs(AM& a,AM& b)
{
memset(dep,0,sizeof(dep));
int h=0,t=1; qa[0]=a.rot; qb[0]=b.rot; dep[qa[0]][qb[0]]=1;
while(h!=t)
{
int ca=qa[h],cb=qb[h]; ++h;
for(int j=0;j<S;j++)
{
int _ca=a.ch[ca][j],_cb=b.ch[cb][j];
if(dep[_ca][_cb]||!_ca) continue;
if(!_cb) return dep[ca][cb];
dep[_ca][_cb]=dep[ca][cb]+1;
qa[t]=_ca; qb[t]=_cb; ++t;
}
}
return -1;
}
#define prt(x) printf("%d\n",x)
char A[SZ],B[SZ];
int main()
{
scanf("%s%s",A,B);
for(int i=0;A[i];i++) SeqA.ins(A[i]-'a'), SufA.ins(A[i]-'a');
for(int i=0;B[i];i++) SeqB.ins(B[i]-'a'), SufB.ins(B[i]-'a');
SeqA.getcnt(); SeqB.getcnt();
SufA.getcnt(); SufB.getcnt();
int AseqBseq=bfs(SeqA,SeqB);
int AseqBsuf=bfs(SeqA,SufB);
int AsufBseq=bfs(SufA,SeqB);
int AsufBsuf=bfs(SufA,SufB);
prt(AsufBsuf);
prt(AsufBseq);
prt(AseqBsuf);
prt(AseqBseq);
}
后缀自动机&序列自动机综合的更多相关文章
- BZOJ4032 [HEOI2015]最短不公共子串 【后缀自动机 + 序列自动机 + dp】
题目链接 BZOJ4032 题解 首先膜\(hb\) 空手切神题 一问\(hash\),二问枚举 三问\(trie\)树,四问\(dp\) 南二巨佬神\(hb\) 空手吊打自动机 \(orz orz ...
- 后缀自动机/回文自动机/AC自动机/序列自动机----各种自动机(自冻鸡) 题目泛做
题目1 BZOJ 3676 APIO2014 回文串 算法讨论: cnt表示回文自动机上每个结点回文串出现的次数.这是回文自动机的定义考查题. #include <cstdlib> #in ...
- bzoj 4032 [ HEOI 2015 ] 最短不公共子串 —— 后缀自动机+序列自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4032 序列自动机其实就是每个位置记录一下某字母后面第一个出现位置,为了子序列能尽量长. 对字 ...
- BZOJ4032: [HEOI2015]最短不公共子串(后缀自动机+序列自动机)
题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...
- 【BZOJ4032】[HEOI2015]最短不公共子串(后缀自动机,序列自动机)
[BZOJ4032][HEOI2015]最短不公共子串(后缀自动机,序列自动机) 题面 BZOJ 洛谷 题解 数据范围很小,直接暴力构建后缀自动机和序列自动机,然后直接在两个自动机上进行\(bfs\) ...
- BZOJ4032[HEOI2015]最短不公共子串——序列自动机+后缀自动机+DP+贪心
题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...
- bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)
bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...
- luoguP4112 [HEOI2015]最短不公共子串 SAM,序列自动机,广搜BFS
luoguP4112 [HEOI2015]最短不公共子串 链接 luogu loj 思路 子串可以用后缀自动机,子序列可以用序列自动机. 序列自动机是啥,就是能访问到所有子序列的自动机. 每个点记录下 ...
- Subsequence(序列自动机模板题)
题目链接:https://nanti.jisuanke.com/t/38232 题目大意:给你一个字符串,然后再给你m个字符串,然后问你在第一个字符串中不连续的子串能不能构成输入的子串. 具体思路:构 ...
随机推荐
- xmpp整理笔记:环境的快速配置(附安装包)
现在虽然环信的xmpp框架很火,但是也有一些弊端.环信的框架部分代码不开源,而且收费模式不科学,用户量一直低于免费线则好,一旦超过,收费极高. xmpp感觉还是从xmppFramework框架学起比较 ...
- 小谈KVC中KeyPath的集合运算符
由于知识点比较简单,这里不再陈述一大堆的原理,直入主题. KVC中的集合运算符有以下三类: 1.简单集合运算符:@avg.@sum.@max.@min.@count (只能用在集合对象中,对象属性必须 ...
- System.Web.HttpException: 无法序列化会话状态。在“StateServer”或“SQLServer”模式下,ASP.NET 将序列化会话状态对象,因此不允许使用无法序列化的对象或 MarshalByRef 对象。如果自定义会话状态存储在“Custom”模式下执行了类似的序列化,则适用同样的限制。 ---> System.Runtime.Serialization.Seria
序列化问题:查询度娘各种答案不一 多为修改web.config找到SessionState节,将Mode 设置为 InProc ,还有舍弃Session用cookie存储,多经试验觉得还是写一个 ...
- 工作中碰到的js问题(disabled表单元素不能提交到服务器)
今天碰到一个奇葩的问题,asp页面表单提交后,有一个文本框<input type="text" name="phone" id="phone&q ...
- 【读书笔记】iOS-ARC-不要向已经释放的对象发送消息
一,在AppDelegate.m中写入如下代码: - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOpti ...
- js DOM Document类型
JavaScript通过Document类型访问文档.在浏览器中,document对象是HTMLDocument(继承自 Document类型)的一个实例,表示整个HTML页面.document对象是 ...
- 我参加了51CTO博客大赛,求投票!
我是张传波,也是Fireball(火球). 我参加了51CTO博客大赛,距离网络投票截止没有几天了,求投票!我的参赛链接:http://blog.51cto.com/contest2013/82313 ...
- android 进程间通信数据(一)------parcel的起源
关于parcel,我们先来讲讲它的“父辈” Serialize. Serialize 是java提供的一套序列化机制.但是为什么要序列化,怎么序列化,序列化是怎么做到的,我们将在本文探讨下. 一:ja ...
- SAE上传文件到storage
还有什么比代码更清晰的讲解 html代码: 一定需要下面这个: method="post" enctype="multipart/form-data" < ...
- 旧项目如何切换到Entity Framework Code First
Entity Framework Code First固然是好东西,然而如果是已经存在的旧有项目,如何简单方便的使用切换呢? 这里介绍一个VS的插件Entity Framework Power Too ...