传送门

Description

​ 给你前alphabet个小写字母组成的字符集, 以及n个单词, 定义一个串s的禁忌值为 \(\sum_{i } [s[i] == Taboo[i]]\) , Taboo[i]为第i个单词,现在给定一个长度len,求随机一个用字符集生成的len长度的串的禁忌值的期望.

Solution

​ 听说出题人卡double的精度,所以只能用long double

​ 考虑一个当串给定时如何求禁忌值, 我们将所有单词投影到数轴上, 这就变成一个贪心, 即取尽量多的线段, 使他们互不相交. 因此我们只要建立一个AC自动机, 然后在自动机上走,碰到一个单词节点就返回根节点,并且ANS++,就可以模拟贪心的过程.(因为我们贪心是按照左端点排序,所以先在AC自动机里匹配的一定是左端点靠前的串).

​ 考虑一个经典问题:给定一个图,求从u到v刚好经过m条边的路径条数.(n <= 100, m <= 1e9)

​ 结论:只要把邻接矩阵处理出来,然后求它的n次方后的矩阵中, \(a[u][v]\)的值.

​ 证明:考虑只经过一条边, 那么直接输出\(a[u][v]\)即可,在两条边的情况中,有 \(ans = \sum_{i = 1}^{n} a[u][i] * a[i][v]\)

只要枚举u到v的中转点,然后计算即可. 可以发现这正好是矩阵乘法的定义. 那么求邻接矩阵的k次幂就是两点k步的路径条数.

​ 所以用\(a[u][v]\)设为u一步到v的期望, 那么邻接矩阵自乘k次后也就是u到v走k步的期望值.

​ 我们把AC自动机建出来,并设定一个终点,然后每走到一个节点就设定当前节点到它所有儿子的一步期望为\(1.0/alphabet\), 走到一个标有单词的节点.我们就设定它到根节点与终点的一步期望为\(1.0/alphabet\),然后矩阵乘法即可.

​ 要注意AC自动机上如果一个节点的fail节点是单词节点,那么他本身也是单词节点.

​ 这样我们就不用跳出一个点的子树, 并且也不会给fail的子树加期望加重复.

Codes

#include<bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for(int i = (a), i##_end_ = (b); i <= i##_end_; ++i)
#define drep(i, a, b) for(int i = (a), i##_end_ = (b); i >= i##_end_; --i)
#define clar(a, b) memset((a), (b), sizeof(a))
typedef long long LL;
typedef long double LD;
int read() {
int x = 0, flag = 1;
char ch = getchar();
while(!isdigit(ch)) {
if(ch == '-') flag *= -1;
ch = getchar();
}
while(isdigit(ch)) {
x = (x << 3) + (x << 1) + ch - 48;
ch = getchar();
}
return x * flag;
}
void write(LL a) {
if(a >= 10) write(a / 10);
putchar(a % 10 + '0');
} #define Maxn 6
#define Maxc 109
struct matrix {
LD data[Maxc][Maxc];
matrix operator * (const matrix b) const {
matrix c;
rep(i, 0, Maxc - 1)
rep(j, 0, Maxc - 1) {
c.data[i][j] = 0;
rep(k, 0, Maxc - 1)
c.data[i][j] += (*this).data[i][k] * b.data[k][j];
}
return c;
}
matrix operator ^ (int times) const {
matrix res, base = (*this);
rep(i, 0, Maxc - 1)
rep(j, 0, Maxc - 1) res.data[i][j] = (i == j);
while(times) {
if(times & 1) res = res * base;
base = base * base;
times >>= 1;
}
return res;
}
}a, b;
#define Maxl ((Maxn) * 16 * 1009)
int vis[Maxl];
int n, len, alphabet;
namespace AC_DFA {
int v[Maxl], t[Maxl][26], cnt = 1, fail[Maxl];
void insert(char str[]) {
int len = strlen(str), cpos = 1;
rep(i, 0, len - 1) {
int id = str[i] - 'a';
if(!t[cpos][id]) t[cpos][id] = ++cnt;
cpos = t[cpos][id];
}
v[cpos] = 1;
}
queue <int> q;
void get_fail() {
q.push(1);
while(!q.empty()) {
int u = q.front(); q.pop();
v[u] |= v[fail[u]];
int pos;
rep(i, 0, alphabet - 1) {
pos = fail[u];
while(pos && !t[pos][i]) pos = fail[pos];
if(t[u][i]) {
fail[t[u][i]] = pos ? t[pos][i] : 1;
q.push(t[u][i]);
}else t[u][i] = pos ? t[pos][i] : 1;
}
}
}
void make_InitalMatrix() {
while(!q.empty()) q.pop();
q.push(1), vis[1] = 1;
while(!q.empty()) {
int u = q.front(); q.pop();
rep(i, 0, alphabet - 1) {
if(!vis[t[u][i]]) vis[t[u][i]] = 1, q.push(t[u][i]);
if(v[t[u][i]]) {
a.data[u][cnt + 1] += (LD)1.0 / alphabet;
a.data[u][1] += (LD)1.0 / alphabet;
} else a.data[u][t[u][i]] += (LD)1.0 / alphabet;
}
}
a.data[cnt + 1][cnt + 1] = 1;
}
};
char s[Maxn][16];
int main() {
n = read(), len = read(), alphabet = read();
rep(i, 1, n) scanf("%s", s[i]), AC_DFA :: insert(s[i]);
AC_DFA :: get_fail();
AC_DFA :: make_InitalMatrix();
b = a ^ len;
printf("%.6Lf\n", b.data[1][AC_DFA :: cnt + 1]);
return 0;
}

BZOJ2553 [BJWC2011]禁忌的更多相关文章

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

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

  2. BZOJ2553: [BeiJing2011]禁忌

    2553: [BeiJing2011]禁忌 Time Limit: 20 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 203  Solved: ...

  3. BZOJ2553[BeiJing2011]禁忌——AC自动机+概率DP+矩阵乘法

    题目描述 Magic Land上的人们总是提起那个传说:他们的祖先John在那个东方岛屿帮助Koishi与其姐姐Satori最终战平.而后,Koishi恢复了读心的能力…… 如今,在John已经成为传 ...

  4. BZOJ2553 Beijing2011禁忌(AC自动机+动态规划+矩阵快速幂+概率期望)

    考虑对一个串如何分割能取得最大值.那么这是一个经典的线段覆盖问题,显然每次取右端点尽量靠前的串.于是可以把串放在AC自动机上跑,找到一个合法串后就记录并跳到根. 然后考虑dp.设f[i][j]表示前i ...

  5. BZOJ2553 [BeiJing2011]禁忌 【AC自动机 + dp + 矩乘优化】

    题目链接 BZOJ2553 题解 话说在前,此题卡精度,最好开long double 先建\(AC\)自动机 求期望,逆着求,设\(f[i][j]\)为长度为\(i\)的串,当前匹配AC自动机\(j\ ...

  6. [BZOJ2553][BeiJing2011]禁忌 dp+AC自动机+矩阵快速幂

    2553: [BeiJing2011]禁忌 Time Limit: 20 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 1206  Solved ...

  7. 题解 洛谷 P4569 【[BJWC2011]禁忌】

    考虑用\(AC\)自动机来解决本题这样的多字符串匹配问题. 要最大化魔法分割后得到的禁忌串数目,最优情况肯定为在一个串中每个禁忌串的右端点进行分割.对应到\(AC\)自动机上,就是匹配到一个禁忌串后, ...

  8. BJWC2011 禁忌

    题目链接 题解 多模式匹配首先建 AC 自动机,看到 \(len \le 10^9\) 想到矩阵乘法优化. 朴素 DP 关于分割的最大值,可以贪心,只要走到一个能匹配串的点立刻返回根继续匹配就行,一定 ...

  9. [BJWC2011]禁忌 AC 自动机 概率与期望

    #include<cstdio> #include<algorithm> #include<cstring> #include<string> #inc ...

随机推荐

  1. zedboard硬件连接过程

    1.      ZedBoard – Connect a 2nd micro-USBcable between the host machine and connector J17 (JTAG) 2. ...

  2. 搜索引擎keyword智能提示的一种实现

    问题背景 搜索关键字智能提示是一个搜索应用的标配.主要作用是避免用户输入错误的搜索词,并将用户引导到相应的关键词上,以提升用户搜索体验. 美团CRM系统中存在数以百万计的商家,为了让用户高速查找到目标 ...

  3. 看懂JSP声明的格式。。。

    在WebRoot下新建test3.jsp 改动body内容: <%! int a = 3; %>     <% int b = 3; %>     <%= a--  %& ...

  4. properties文件读取配置信息

    public static void main(String[] args){ String printerName=""; String path = "C:\\Bar ...

  5. WIN7 不用格式化磁盘怎么把FAT32系统改成NTFS系统

    开始-运行,输入cmd回车.假设你要转换D盘.输入convert d: /fs:NTFS回车. [ 此时可能会提示: 访问被拒绝 因为你没有足够的特权 是权限不够的原因 开始--程序--附件 右键&q ...

  6. 头文件与cpp文件为什么要分开写

    最表面的机制是: 头文件是程序的界面(是代码界面),提供给程序员以  类.模版.函数等一系列的声明,让程序员知道应该怎么调用里面的"东西". 从动态链接库的角度看: 头文件提供界面 ...

  7. qrcode.react和jquery.qrcode生成二维码

    qrcode.react 1.安装 npm install qrcode.react 2.用法(这里用的ant design) import React from 'react'; import QR ...

  8. 关于warning: Clock skew detected. Your build may be incomplete. 的解决方法【转】

    本文转载自:http://blog.csdn.net/jeesenzhang/article/details/40300127 今天发现电脑的系统时间不正确,因此将时钟进行了修改,回头编译Linux ...

  9. UVA11383 Golden Tiger Claw —— KM算法

    题目链接:https://vjudge.net/problem/UVA-11383 题解: 根据KM()算法,标杆满足:l(x) + l(y) >= w(x, y) . 当求完最大权匹配之后,所 ...

  10. HDU1166 敌兵布阵 —— 线段树单点修改

    题目链接:https://vjudge.net/problem/HDU-1166 C国的死对头A国这段时间正在进行军事演习,所以C国间谍头子Derek和他手下Tidy又开始忙乎了.A国在海岸线沿直线布 ...