Luogu-3250 [BJOI2017]魔法咒语(AC自动机,矩阵快速幂)

题目链接

题解:

多串匹配问题,很容易想到是AC自动机

先构建忌讳词语的AC自动机,构建时顺便记录一下这个点以及它的所有后缀有没有忌讳词语,即对于每个AC自动机上的结点\(x\),\(p[x].p|=p[p[x].f].p\)

然后前半部分分和后半是两道完全不同的题目(滑稽

前60分:

这些部分分的特征是\(L\le 100\)

直接AC自动机上\(dp\)就好了,枚举匹配长度\(i\),当前匹配到的点\(x\),以及后面要匹配的基本词汇\(s[j]\),找到匹配了这个串后到达的点\(y\),如果匹配过程中没有经过忌讳词语结点,就进行转移\(f[i+len[s[j]]][y]+=f[i][x]\)

最后统计下终点在每个点的情况就好了

inline void work(){
memset(f,0,sizeof(f));
f[0][0]=1;
for(int i=0;i<L;i++)
for(int x=0;x<=tot;x++){
for(int j=1;j<=n;j++){
int len=strlen(a[j]+1);
if(i+len>L) continue;
int y=run(x,a[j]);
if(y==-1) continue;
(f[i+len][y]+=f[i][x])%=P;
}
}
int Ans=0;
for(int x=0;x<=tot;x++) (Ans+=f[L][x])%=P;
printf("%d\n",Ans);
}

后40分:

这些测试点的特点就是基本词汇长度小于等于2

\(\sum s[i]\)这么小,\(L\)这么大不禁让人往矩阵快速幂上想

利用矩阵乘法进行转移,如果设被乘的矩阵\(S\)中\(S[0][i]\)代表每个点是否可以在串长为0时匹配(很明显只有\(S[0][0]=1\))。那么如果我们构建乘它的矩阵为\(G\),根据矩阵乘法的运算法则:

\[T[0][j]=\sum_{k=0}^n{S[0][k]*G[k][j]}
\]

可以发现,如果\(G[x][y]\)代表的是走一步从\(x\)转移到\(y\)的方案数,得到的矩阵\(T\)中的元素\(T[0][x]\)就是走一步转移到\(x\)的方案数,也就是相当于模拟走了一步。如果我们走了\(L\)步让\(S*G^L\)就好了,矩阵乘法有结合律,我们就可以先用快速幂算出\(G^L\),最后再乘\(S\)

但是这样做只能处理串长为\(1\)的情况,因为没有保留前一步的方案数,串长为\(2\)的转移是无法处理的。也就是说,设当前是第\(k\)步,转移应该包括两部分:一部分是从\(S_k\)走一步到\(S_{k+1}\),一部分是从\(S_{k-1}\)走两步到\(S_{k+1}\)。

这种需要同时考虑这一步和上一步的矩阵要如何转移呢?

举一个非常简单的例子吧:求斐波那契数列的第\(n\)项。

由于\(Fib[i+1]=Fib[i-1]+Fib[i]\),我们就用了一个大小为\(1*2\)的矩阵\(S\),\(S[0][0]\)记录\(Fib[i-1]\),\(S[0][1]\)记录\(Fib[i]\)。

转移矩阵\(G\)是这样的(矩阵下标从零开始):

\[G=
\left[
\begin{matrix}
0&1\\
1&1
\end{matrix}
\right]
\]

联系\(S[0][0]与S[0][1]\)的含义和矩阵乘法的含义:

\(G[1][0]=1\)代表下一步的\(Fib[i-1]\)就是这一步的\(Fib[i]\)

\(G[1][1]=1\)代表下一步的\(Fib[i]\)能由这一步的\(Fib[i]\)转移

\(G[0][1]=1\)代表下一步的\(Fib[i]\)能由这一步的\(Fib[i-1]\)转移

\(G[0][0]=0\)是因为下一步的\(Fib[i-1]\)已经钦定为这一步的\(Fib[i]\),无需其他转移

总的来说,转移矩阵包括四部分:这一个到下一个(走一步),前一个到下一个(走两步),这一个到“下一个的前一个”(单位矩阵),前一个到下一个的前一个(一般为全\(0\))

了解了矩阵乘法求\(Fib[n]\)的原理之后,这道题就很简单了

构造大小为\(2*n\)的\(S\),\(S[0][0\sim n]\)记前一步的方案数,\(S[0][n+1\sim n+1+n]\)记这一步的方案数

构造大小为\(2n*2n\)的\(G\),右下是长为\(1\)串的转移矩阵,右上是长为\(2\)串的转移矩阵,左下是单位矩阵,左上是全\(0\)矩阵

最后得到的矩阵\(T=S*G^L\)中,\(T[0][n+1\sim n+1+n]\)的和即为答案。

代码:

  • 处理长度分别为\(1,2\)的串的转移
for(int x=0;x<=tot;x++){
for(int i=1;i<=n;i++){
if(strlen(a[i]+1)>1) continue;
int y=run(x,a[i]);
if(y==-1) continue;
g.a[tot+1+x][tot+1+y]++;
}
}
for(int x=0;x<=tot;x++){
for(int i=1;i<=n;i++){
if(strlen(a[i]+1)<2) continue;
int y=run(x,a[i]);
if(y==-1) continue;
g.a[x][tot+1+y]++;
}
}
  • 左下角单位矩阵
	for(int x=0;x<=tot;x++)
g.a[tot+1+x][x]++;
  • 矩阵快速幂
	g=poww(g,L);
int Ans=0;
for(int x=0;x<=tot;x++)
(Ans+=g.a[tot+1][tot+1+x])%=P;
printf("%d\n",(Ans+P)%P);

代码:

#include<map>
#include<set>
#include<cmath>
#include<ctime>
#include<queue>
#include<stack>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define qmax(x,y) (x=max(x,y))
#define qmin(x,y) (x=min(x,y))
#define mp(x,y) make_pair(x,y)
using namespace std;
typedef long long ll;
typedef pair<int,int> pii;
inline int read(){
int ans=0,fh=1;
char ch=getchar();
while(ch<'0'||ch>'9'){
if(ch=='-') fh=-1;
ch=getchar();
}
while(ch>='0'&&ch<='9')
ans=ans*10+ch-'0',ch=getchar();
return ans*fh;
}
const int maxn=150,P=1e9+7;
struct node{
int son[26],p,f;
}p[maxn];
struct matrix{
int a[210][210];
}g,base,tmp;
int tot,n,m,L,f[200][maxn];
char a[60][maxn],b[60][maxn];
matrix operator * (matrix x,matrix y){
for(int i=0;i<=tot+1+tot;i++)
for(int j=0;j<=tot+1+tot;j++){
tmp.a[i][j]=0;
for(int k=0;k<=tot+1+tot;k++)
(tmp.a[i][j]+=1ll*x.a[i][k]*y.a[k][j]%P)%=P;
}
return tmp;
}
inline void insert(char *s){
int len=strlen(s+1),x=0;
for(int i=1;i<=len;i++){
int z=s[i]-'a';
if(!p[x].son[z])
p[x].son[z]=++tot;
x=p[x].son[z];
}
p[x].p=1;
}
queue<int>q;
inline void build(){
for(int i=0;i<26;i++)
if(p[0].son[i]) q.push(p[0].son[i]);
while(!q.empty()){
int x=q.front();q.pop();
for(int i=0;i<26;i++)
if(p[x].son[i]){
p[p[x].son[i]].f=p[p[x].f].son[i];
q.push(p[x].son[i]);
}
else p[x].son[i]=p[p[x].f].son[i];
p[x].p|=p[p[x].f].p;
}
}
inline int run(int x,char *s){
int len=strlen(s+1);
for(int i=1;i<=len;i++){
int z=s[i]-'a';
x=p[x].son[z];
if(p[x].p) return -1;
}
return x;
}
inline void work(){
memset(f,0,sizeof(f));
f[0][0]=1;
for(int i=0;i<L;i++)
for(int x=0;x<=tot;x++){
for(int j=1;j<=n;j++){
int len=strlen(a[j]+1);
if(i+len>L) continue;
int y=run(x,a[j]);
if(y==-1) continue;
(f[i+len][y]+=f[i][x])%=P;
}
}
int Ans=0;
for(int x=0;x<=tot;x++) (Ans+=f[L][x])%=P;
printf("%d\n",Ans);
}
inline matrix poww(matrix x,int y){
for(int i=0;i<=tot+1+tot;i++)
base.a[i][i]=1;
while(y){
if(y&1) base=base*x;
x=x*x,y>>=1;
}
return base;
}
inline void work2(){
for(int x=0;x<=tot;x++){
for(int i=1;i<=n;i++){
if(strlen(a[i]+1)>1) continue;
int y=run(x,a[i]);
if(y==-1) continue;
g.a[tot+1+x][tot+1+y]++;
}
}
for(int x=0;x<=tot;x++){
for(int i=1;i<=n;i++){
if(strlen(a[i]+1)<2) continue;
int y=run(x,a[i]);
if(y==-1) continue;
g.a[x][tot+1+y]++;
}
}
for(int x=0;x<=tot;x++)
g.a[tot+1+x][x]++;
g=poww(g,L);
int Ans=0;
for(int x=0;x<=tot;x++)
(Ans+=g.a[tot+1][tot+1+x])%=P;
printf("%d\n",(Ans+P)%P);
}
int main(){
// freopen("nh.in","r",stdin);
// freopen("zhy.out","w",stdout);
n=read(),m=read(),L=read();
for(int i=1;i<=n;i++) scanf("%s",a[i]+1);
for(int i=1;i<=m;i++) scanf("%s",b[i]+1);
for(int i=1;i<=m;i++) insert(b[i]);
build();
if(L<=100) work();
else work2();
return 0;
}

Luogu-3250 [BJOI2017]魔法咒语(AC自动机,矩阵快速幂)的更多相关文章

  1. [BZOJ4861][BJOI2017]魔法咒语(AC自动机+矩阵优化DP)

    4861: [Beijing2017]魔法咒语 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 217  Solved: 105[Submit][Sta ...

  2. POJ2778 DNA Sequence(AC自动机+矩阵快速幂)

    题目给m个病毒串,问不包含病毒串的长度n的DNA片段有几个. 感觉这题好神,看了好久的题解. 所有病毒串构造一个AC自动机,这个AC自动机可以看作一张有向图,图上的每个顶点就是Trie树上的结点,每个 ...

  3. poj2778DNA Sequence (AC自动机+矩阵快速幂)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud DNA Sequence Time Limit: 1000MS   Memory ...

  4. HDU 2243考研路茫茫——单词情结 (AC自动机+矩阵快速幂)

    背单词,始终是复习英语的重要环节.在荒废了3年大学生涯后,Lele也终于要开始背单词了. 一天,Lele在某本单词书上看到了一个根据词根来背单词的方法.比如"ab",放在单词前一般 ...

  5. POJ2778(SummerTrainingDay10-B AC自动机+矩阵快速幂)

    DNA Sequence Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 17160   Accepted: 6616 Des ...

  6. poj2778 ac自动机+矩阵快速幂

    给m个子串,求长度为n的不包含子串的母串数,最直接的应该是暴搜,肯定tle,考虑用ac自动机 将子串建成字典树,通过next表来构造矩阵,然后用矩阵快速幂求长度为n的数量 邻接矩阵https://we ...

  7. HDU 2243 考研路茫茫――单词情结 ——(AC自动机+矩阵快速幂)

    和前几天做的AC自动机类似. 思路简单但是代码200余行.. 假设solve_sub(i)表示长度为i的不含危险单词的总数. 最终答案为用总数(26^1+26^2+...+26^n)减去(solve_ ...

  8. POJ - 2778 ~ HDU - 2243 AC自动机+矩阵快速幂

    这两题属于AC自动机的第二种套路通过矩阵快速幂求方案数. 题意:给m个病毒字符串,问长度为n的DNA片段有多少种没有包含病毒串的. 根据AC自动机的tire图,我们可以获得一个可达矩阵. 关于这题的t ...

  9. 考研路茫茫——单词情结 HDU - 2243 AC自动机 && 矩阵快速幂

    背单词,始终是复习英语的重要环节.在荒废了3年大学生涯后,Lele也终于要开始背单词了. 一天,Lele在某本单词书上看到了一个根据词根来背单词的方法.比如"ab",放在单词前一般 ...

  10. POJ 2778 DNA Sequence(AC自动机 + 矩阵快速幂)题解

    题意:给出m个模式串,要求你构造长度为n(n <= 2000000000)的主串,主串不包含模式串,问这样的主串有几个 思路:因为要不包含模式串,显然又是ac自动机.因为n很大,所以用dp不太好 ...

随机推荐

  1. Android开发:《Gradle Recipes for Android》阅读笔记(翻译)4.5——使用Android Libraries

    问题: 你想要在app当中增加新的library模块 解决方案: 使用library插件,增加一个library模块作为依赖. 讨论: 不可以通过使用java库给app增加许多功能,通常是使用jar包 ...

  2. Castle 整合.NET Remoting

    今天研究了一下Castle的Remoting Facility.记录如下: 微软以前使用COM/DCOM的技术来处理分布式系统架构,通过Client端的Proxy代理程序来呼叫远程Server机器上的 ...

  3. 关于vs2013中包含目录,以及库目录配置相对路径的问题

    记住一句话即可! 相对路径: 是相对于你的工程的*.vcxproj的路径!!!

  4. Manacher模板,kmp,扩展kmp,最小表示法模板

    *N]; //储存临时串 *N];//中间记录 int Manacher(char tmp[]) { int len=strlen(tmp); ; ;i<len;i++) { s[cnt++]= ...

  5. 【BZOJ2090/2089】[Poi2010]Monotonicity 2 动态规划+线段树

    [BZOJ2090/2089][Poi2010]Monotonicity Description 给出N个正整数a[1..N],再给出K个关系符号(>.<或=)s[1..k].选出一个长度 ...

  6. mysql source 执行sql脚本,中文变量不显示问题或乱码问题

    执行脚本内容如下: SET @pre_version=2017080901; SET @cur_version=2017090401; SET @ver_desc = '测试脚本'; CALL pro ...

  7. IO流入门-第九章-BufferedReader_BufferedWriter复制

    利用BufferedReader和BufferedWriter进行复制粘贴 import java.io.*; public class BufferedReader_BufferedWriterCo ...

  8. Python--(并发编程之线程Part2)

    GIL只能保证垃圾回收机制的安全,进程中的数据安全还是需要自定义锁 线程执行代码首先要抢到GIL全局锁,假设线程X首先抢到,以为要抢到自定义锁要执行代码,所以这个线程在执行代码的时候就很容抢到了自定义 ...

  9. 003-基于URL的权限管理[不使用shiro]

    一.基于url权限管理流程[实现步骤] 基于url拦截是企业中常用的权限管理方法,实现思路是:将系统操作的每个url配置在权限表中,将权限对应到角色,将角色分配给用户,用户访问系统功能通过Filter ...

  10. Jupyter Notebook修改目标文件

    默认的路径 如果没有修改配置文件,那么一般就在用户目录下面: 下面各处默认起始目标地址,以防有一天想改回来 I:\shujufenxi\python.exe I:\shujufenxi\cwp.py ...