P3715 [BJOI2017]魔法咒语

用基本词汇组成\(L\)长度的单词,其中不能包含禁忌词汇

用禁忌词汇建强大的\(tire\)图

解决:

分类讨论,\(L<=100\)用普通dp暴力在\(tire\)图上用基本词汇跑,不能包含禁忌词汇

显然,\(L<=10^8\)肯定会超时的,注意到基本词汇的长度\(<=2\),矩阵优化一下

构造矩阵是一个难点,左边的矩阵两部分,我们要把\(dp_{i-2}与dp_{i-1}\)矩乘后达到\(dp_{i-1}与dp_{i}\)

右边的矩阵四部分

\((1)空 (2)len=2\)

$(3)单位矩阵(4)len=1 $

构造到此为止 直接快速幂就好了

My complete code

#include<bits/stdc++.h>
#include<queue>
using namespace std;
typedef long long LL;
const LL maxn=1010;
const LL MOD=1e9+7;
LL n,m,L,nod=1,ans;
LL son[maxn][30],end[maxn],fail[maxn],len[maxn],dp[maxn][maxn];
char ji[maxn][maxn];
inline void Insert(char *s){
LL l=strlen(s),now=1;
for(LL i=0;i<l;++i){
LL c=s[i]-'a';
if(!son[now][c])
son[now][c]=++nod;
now=son[now][c];
}
end[now]=true;
}
inline void Fbuild(){
queue<LL> que;
for(LL i=0;i<26;++i)
if(son[1][i]){
que.push(son[1][i]);
fail[son[1][i]]=1;
}else
son[1][i]=1;
while(que.size()){
LL u=que.front(); que.pop();
for(LL i=0;i<26;++i){
LL v=son[u][i];
if(v){
fail[v]=son[fail[u]][i];
que.push(v);
end[v]|=end[fail[v]];
}else
son[u][i]=son[fail[u]][i];
}
}
}
struct mat{
LL m[300][300];
}a,b,r;
inline LL Go(LL now,LL c){
return (end[son[now][c]])?-1:son[now][c];
}
inline mat Mul(const mat &x,const mat &y){
mat res;
memset(res.m,0,sizeof(res.m));
for(LL i=1;i<=2*nod;++i)
for(LL j=1;j<=2*nod;++j)
for(LL k=1;k<=2*nod;++k)
res.m[i][j]=(res.m[i][j]+x.m[i][k]*y.m[k][j]%MOD)%MOD;
return res;
}
inline void Pow(LL mi){
while(mi){
if(mi&1)
r=Mul(r,b);
b=Mul(b,b);
mi>>=1;
}
}
int main(){
scanf("%lld%lld%lld",&n,&m,&L);
for(LL i=1;i<=n;++i){
scanf(" %s",ji[i]);
len[i]=strlen(ji[i]);
}
for(LL i=1;i<=m;++i){
char s[maxn];
scanf(" %s",s);
Insert(s);
}
Fbuild();
if(L<=100){
dp[0][1]=1;
for(LL i=0;i<L;++i)
for(LL j=1;j<=nod;++j){
if(!dp[i][j])
continue;
for(LL k=1;k<=n;++k){//沿着基本串走
if(i+len[k]>L)
continue;
LL now=j;
for(LL q=0;q<len[k]&&now!=-1;++q)
now=Go(now,ji[k][q]-'a');
if(now!=-1)
dp[i+len[k]][now]=(dp[i+len[k]][now]+dp[i][j])%MOD;
}
}
for(LL i=1;i<=nod;++i)
ans=(ans+dp[L][i])%MOD;
printf("%lld",ans);
return 0;
} a.m[1][1+nod]=1;
for(LL i=1;i<=nod;++i)
b.m[i+nod][i]=1;
for(LL i=1;i<=nod;++i)
for(LL j=1;j<=n;++j){
if(len[j]!=1)
continue;
LL now=i;
if(end[now])
continue;
now=Go(now,ji[j][0]-'a');
if(now!=-1)
++b.m[nod+i][nod+now];
}
for(LL i=1;i<=nod;++i)
for(LL j=1;j<=n;++j){
if(len[j]!=2)
continue;
LL now=i;
if(end[now])
continue;
now=Go(now,ji[j][0]-'a');
if(now==-1)
continue;
now=Go(now,ji[j][1]-'a');
if(now!=-1)
++b.m[i][nod+now];
}
for(LL i=1;i<=2*nod;++i)
r.m[i][i]=1;
Pow(L);
a=Mul(a,r);
for(LL i=1;i<=nod;++i)
ans=(ans+a.m[1][nod+i])%MOD;
printf("%lld",ans);
return 0;
}

P3715 [BJOI2017]魔法咒语的更多相关文章

  1. bzoj4861 / P3715 [BJOI2017]魔法咒语

    P3715 [BJOI2017]魔法咒语 AC自动机+dp+矩阵乘法 常规思路是按基本串建立AC自动机 然鹅这题是按禁忌串建立AC自动机 对后缀是禁忌的点以及它的失配点做上标记$(a[i].ed)$, ...

  2. [BJOI2017]魔法咒语 --- AC自动机 + 矩阵优化

    bzoj 4860   LOJ2180   洛谷P3175 [BJOI2017]魔法咒语 题目描述: Chandra 是一个魔法天才. 从一岁时接受火之教会洗礼之后,Chandra 就显示出对火元素无 ...

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

    Luogu-3250 [BJOI2017]魔法咒语(AC自动机,矩阵快速幂) 题目链接 题解: 多串匹配问题,很容易想到是AC自动机 先构建忌讳词语的AC自动机,构建时顺便记录一下这个点以及它的所有后 ...

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

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

  5. 【[BJOI2017]魔法咒语】

    矩阵乘法+\(AC\)自动机 是道很不错的题了 首先是前六十分,就是一个\(AC\)自动机上的套路\(dp\),设\(dp[i][j]\)表示匹配出的长度为\(i\)在自动机上位置为\(j\)的方案数 ...

  6. [BJOI2017]魔法咒语

    Description Chandra 是一个魔法天才. 从一岁时接受火之教会洗礼之后, Chandra 就显示出对火元素无与伦比的亲和力,轻而易举地学会种种晦涩难解的法术.这也多亏 Chandra ...

  7. AHOI2018训练日程(3.10~4.12)

    (总计:共90题) 3.10~3.16:17题 3.17~3.23:6题 3.24~3.30:17题 3.31~4.6:21题 4.7~4.12:29题 ZJOI&&FJOI(6题) ...

  8. AC 自动机学习笔记

    虽然 NOIp 原地爆炸了,目前进入 AFO 状态,但感觉省选还是要冲一把,所以现在又来开始颓字符串辣 首先先复习一个很早很早就学过但忘记的算法--自动 AC AC自动机. AC 自动机能够在 \(\ ...

  9. 【题解】AC自动机题解合集

    最近貌似大家都在搞字符串?很长一段时间都没有写博客了……还是补一补坑吧. 感觉AC自动机真的非常优美了,通过在trie树上建立fail指针可以轻松解决多模匹配的问题.实际上在AC自动机上的匹配可以看做 ...

随机推荐

  1. JAVA实现EXCEL公式专题(四)——字符串函数

    直接上代码: /** * 项目名称: * 文件说明: ExCEL公式类型:字符串公式 * 主要特点: * 版本:1.0 * 制作人:刘晨曦 * 创建时间:2013-12-3 **/ package E ...

  2. MyBatis-Invalid bound statement (not found)-问题处理

    最近把工程改为Hibernate和MyBatis并存,并存只要注意两点即可: 1.使用同一个dataSource 2.事物交给Hibernate进行管理(Hibernate4+)  Hibernate ...

  3. spark-submit提交方式测试Demo

    写一个小小的Demo测试一下Spark提交程序的流程 Maven的pom文件 <properties> <maven.compiler.source>1.7</maven ...

  4. 【前端阅读】——《JavaScript入门经典》摘记之JavaScript与XML

    前言:这本书除了基础的JavaScript理论体系之外,有一个特别的章节,就是讲解——JavaScript与XML的关系,从中,我更进一步的了解了XML的基础.创建.显示以及使用JavaScript如 ...

  5. Leetcode 编程训练(转载)

    Leetcode这个网站上的题都是一些经典的公司用来面试应聘者的面试题,很多人通过刷这些题来应聘一些喜欢面试算法的公司,比如:Google.微软.Facebook.Amazon之类的这些公司,基本上是 ...

  6. HTTP学习笔记(一)报文和连接管理

    对TCP/IP协议簇有些了解的同学们应该都知道.TCP/IP协议通过精简ISO网络7层协议(事实上了解历史渊源的话,TCP/IP协议本来目的并非简化ISO的7层协议.仅仅是因为ISO协议簇制定速度慢于 ...

  7. Sql 复习3之存储管理

    一.事务管理 单个工作单元称为事务,我们将若干条sql命令组合在一起,形成存储过程.触发器等,利用存储过程和触发器可以进行事务管理. 二.编程基础介绍 主要有:函数.程序设计语句等. 程序设计语句: ...

  8. Linux禁止Ctrl+Alt+Del重新启动

    方法1:改动/etc/inittab 屏蔽 ca:12345:ctrlaltdel:/sbin/shutdown -t1 -a -r now 或者删除改行内容 保存退出 适用对象:RedHat4.8 ...

  9. 14:质数因子PrimeNum

    14:题目描述 功能:输入一个正整数,按照从小到大的顺序输出它的所有质数的因子(如180的质数因子为2 2 3 3 5 ) 详细描述: 函数接口说明: public String getResult( ...

  10. MySql(八):MySQL性能调优——Query 的优化

    一.理解MySQL的Query Optimizer MySQL Optimizer是一个专门负责优化SELECT 语句的优化器模块,它主要的功能就是通过计算分析系统中收集的各种统计信息,为客户端请求的 ...