题面传送门

又好久没做过 AC 自动机的题了,做道练练手罢(

首先考虑对于某个固定的字符串怎样求出它的伤害,我们考虑贪心,每碰到出现一个模式串就将其划分为一段,最终该字符串的代价就是划分的次数。具体来说我们记录一个 \(pre\) 表示上一次在 \(pre\) 与 \(pre-1\) 划分为一段,我们从前往后扫一遍,每扫到一个 \(i\),就检验 \(s[pre...i],s[pre+1...i],s[pre+2...i],\cdots,s[i...i]\) 中是否有子串在模式串中出现过,如果有就令答案加 \(1\),并将 \(pre\) 赋为 \(i\)。正确性显然。

显然这种多模式串的题可以想到 AC 自动机。建出自动机,然后设 \(dp_{i,j}\) 表示已经填好了前 \(i\) 个位置上的字符,当前匹配到了 \(j\) 号节点的概率,转移显然可以枚举 \(j\) 的子节点 \(k\),\(dp_{i+1,k}\leftarrow dp_{i,j}\times\dfrac{1}{\text{alpha}}\),那么怎么体现”划分完的部分就不考虑了“呢?很简单,如果 \(k\) 是匹配节点,那我们就令 \(dp_{i+1,\text{root}}\leftarrow dp_{i,j}\times\dfrac{1}{\text{alpha}}\) instead of \(dp_{i+1,k}\leftarrow dp_{i,j}\times\dfrac{1}{\text{alpha}}\),并且由于期望的线性性,这边答案产生了 \(1\) 的贡献,故还需令 \(ans\leftarrow dp_{i,j}\times\dfrac{1}{\text{alpha}}\)。

由于这题 \(len\) 数据范围很大,故可以套路地想到矩阵快速幂。建立转移矩阵,快速幂转移即可。时间复杂度 \((\sum|s|)\log len\)。还有一个小问题,就是这个 \(ans\) 如何一边求 \(dp\) 的值一遍维护。我们考虑令矩阵的大小扩大 \(1\),即一般的矩阵快速幂我们是 \(\begin{bmatrix}dp_{i,1}\\dp_{i,2}\\\cdots\\dp_{i,m}\end{bmatrix}\) 转移到 \(\begin{bmatrix}dp_{i+1,1}\\dp_{i+1,2}\\\cdots\\dp_{i+1,m}\end{bmatrix}\),而这题我们考虑增加一行 \(ans\),即从 \(\begin{bmatrix}dp_{i,1}\\dp_{i,2}\\\cdots\\dp_{i,m}\\ans\end{bmatrix}\) 转移到 \(\begin{bmatrix}dp_{i+1,1}\\dp_{i+1,2}\\\cdots\\dp_{i+1,m}\\ans\end{bmatrix}\)。具体来说对于一个点 \(j\),如果它的儿子中有 \(c\) 个匹配点,那么我们就令 \(ans\leftarrow ans+dp_{i,j}\times\dfrac{c}{\text{alpha}}\),这个也可在转移矩阵中表示出来。

最后再多说一句,如果你矩阵乘法是使用矩阵乘向量,也就是矩阵在前,向量在后的写法,要将转移到的位置设作行号,从哪个位置转移设为列号,我因为这地方写反了调了好久……xtbz

typedef long double ld;
const int MAXL=75;
const int ALPHA=26;
int n,len,alpha;char buf[MAXL+5];
int ch[MAXL+5][ALPHA+2],fail[MAXL+5],ncnt=0;
bool ed[MAXL+5];
void insert(char *s){
int len=strlen(s+1),cur=0;
for(int i=1;i<=len;i++){
if(!ch[cur][s[i]-'a']) ch[cur][s[i]-'a']=++ncnt;
cur=ch[cur][s[i]-'a'];
} ed[cur]=1;
}
void getfail(){
queue<int> q;
for(int i=0;i<alpha;i++) if(ch[0][i]) q.push(ch[0][i]);
while(!q.empty()){
int x=q.front();q.pop();
for(int i=0;i<alpha;i++){
if(ch[x][i]) fail[ch[x][i]]=ch[fail[x]][i],q.push(ch[x][i]);
else ch[x][i]=ch[fail[x]][i];
}
}
}
struct mat{
ld a[MAXL+5][MAXL+5];
mat(){memset(a,0,sizeof(a));}
mat operator *(const mat &rhs){
mat res;
for(int i=0;i<=ncnt;i++) for(int j=0;j<=ncnt;j++)
for(int k=0;k<=ncnt;k++) res.a[i][j]+=a[i][k]*rhs.a[k][j];
return res;
}
};
int main(){
scanf("%d%d%d",&n,&len,&alpha);
for(int i=1;i<=n;i++) scanf("%s",buf+1),insert(buf);
getfail();mat trs,mul,tt;
for(int i=0;i<=ncnt;i++) for(int j=0;j<alpha;j++){
if(ed[ch[i][j]]) trs.a[0][i]+=1.0/alpha,trs.a[ncnt+1][i]+=1.0/alpha;
else trs.a[ch[i][j]][i]+=1.0/alpha;
} ncnt++;trs.a[ncnt][ncnt]=1;
for(int i=0;i<=ncnt;i++) mul.a[i][i]=1;
for(;len;len>>=1,trs=trs*trs) if(len&1) mul=mul*trs;
tt.a[0][0]=1;tt=mul*tt;printf("%.10Lf\n",tt.a[ncnt][0]);
return 0;
}

洛谷 P4569 - [BJWC2011]禁忌(AC 自动机+矩阵乘法)的更多相关文章

  1. bzoj 2553: [BeiJing2011]禁忌 AC自动机+矩阵乘法

    题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=2553 题解: 利用AC自动机的dp求出所有的转移 然后将所有的转移储存到矩阵中,进行矩阵 ...

  2. 【bzoj1444】[Jsoi2009]有趣的游戏 AC自动机+矩阵乘法

    题目描述 输入 注意 是0<=P 输出 样例输入 样例输出 题解 AC自动机+矩阵乘法 先将所有字符串放到AC自动机中,求出Trie图. 然后构建邻接矩阵:如果x不是某个字符串的末位置,则x连向 ...

  3. BZOJ2553 [BeiJing2011]禁忌 AC自动机 矩阵

    原文链接http://www.cnblogs.com/zhouzhendong/p/8196279.html 题目传送门 - BZOJ2553 题意概括 引用一下lych大佬的: 在字母只有前alph ...

  4. [BJOI2011]禁忌 --- AC自动机 + 矩阵优化 + 期望

    bzoj 2553 [BJOI2011]禁忌 题目描述: Magic Land上的人们总是提起那个传说:他们的祖先John在那个东方岛屿帮助Koishi与其姐姐Satori最终战平.而后,Koishi ...

  5. 洛谷-P5357-【模板】AC自动机(二次加强版)

    题目传送门 -------------------------------------- 过年在家无聊补一下这周做的几道AC自动机的模板题 sol:AC自动机,还是要解决跳fail边产生的重复访问,但 ...

  6. 洛谷P3966 [TJOI2013]单词(AC自动机)

    题目描述 小张最近在忙毕设,所以一直在读论文.一篇论文是由许多单词组成但小张发现一个单词会在论文中出现很多次,他想知道每个单词分别在论文中出现了多少次. 输入输出格式 输入格式: 第一行一个整数N,表 ...

  7. 洛谷P3966 单词 [TJOI2013] AC自动机

    正解:AC自动机 解题报告: 传送门! 先来提供一个40pts错解QAQ 首先看到这题就会想到AC自动机板子题2鸭!然后就照着那题的套路打一下,随便改一点儿,简单来说就是每次经过一个节点都要++,然后 ...

  8. 洛谷P2444 病毒 [POI2000] AC自动机

    正解:AC自动机 解题报告: 传送门! 首先看到这种题目二话不说先把trie树和fail指针建立起来 然后就想鸭,如果我们想让模式串和文本串尽量不能匹配,就要想办法让它跳fail指针,而不是继续往下走 ...

  9. 洛谷P2444 病毒【AC自动机】

    题目描述 二进制病毒审查委员会最近发现了如下的规律:某些确定的二进制串是病毒的代码.如果某段代码中不存在任何一段病毒代码,那么我们就称这段代码是安全的.现在委员会已经找出了所有的病毒代码段,试问,是否 ...

随机推荐

  1. Windows内核开发-10-监听对象

    Windows内核开发-10-监听对象 Windows内核除了可以监听进程,线程.dll还可以监听特定的对象和注册表.这里先讲一下监听对象. 监听对象 内核提供了一种可以监听对特定的对象类型的句柄进行 ...

  2. Stream中的Pipeline理解

    使用Stream已经快3年了,但是从未真正深入研究过Stream的底层实现. 今天开始把最近学到的Stream原理记录一下. 本篇文章简单描述一下自己对pipeline的理解. 基于下面一段代码: p ...

  3. 【UE4 C++ 基础知识】<7> 容器——TSet

    概述 TSet是一种快速容器类,(通常)用于在排序不重要的情况下存储唯一元素. TSet 类似于 TMap 和 TMultiMap,但有一个重要区别:TSet 是通过对元素求值的可覆盖函数,使用数据值 ...

  4. SpringMvc 中 FrameworkServlet 覆盖 service 的有点。

    @Override protected void service(HttpServletRequest request, HttpServletResponse response) throws Se ...

  5. 第四次Alpha Scrum Meeting

    本次会议为Alpha阶段第四次Scrum Meeting会议 会议概要 会议时间:2021年4月28日 会议地点:线上会议 会议时长:18min 会议内容简介:本次会议主要由每个人展示自己目前完成的工 ...

  6. [对对子队]会议记录4.12(Scrum Meeting 3)

    今天已完成的工作 朱骏豪 ​ 工作内容:找到了游戏的背景场景,用PS扣了按钮的图 ​ 相关issue:实现UI的美术需求 实现游戏场景中的必要模型 梁河览 ​ 工作内容:将关卡选择界面和欢迎界面导入项 ...

  7. [no code][scrum meeting] Alpha 12

    项目 内容 会议时间 2020-04-19 会议主题 周总结会议 会议时长 45min 参会人员 全体成员 $( "#cnblogs_post_body" ).catalog() ...

  8. BUAA-软件工程-个人总结与心得

    提问回顾以及个人总结 项目 内容 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 提问回顾与个人总结 我在这个课程的目标是 学习软件开发的过程,团队之间的写作 ...

  9. logstash的安装和简单使用

    logstash的安装和简单使用 一.安装 1.下载并解压 2.logstash 一些命令行参数 1.查看帮助信息 2.加载指定pipeline文件路径 3.检测配置文件语法是否有错误 4.热加载pi ...

  10. 计算机网络之IPv4(IPv4分组、IPv4地址、NAT、子网划分与子网掩码、CIDR、ARP协议、DHCP、ICMP)

    文章转自:https://blog.csdn.net/weixin_43914604/article/details/105138313 学习课程:<2019王道考研计算机网络> 学习目的 ...