[BZOJ4820]硬币游戏 KMP+高斯消元

4820: [Sdoi2017]硬币游戏

Time Limit: 10 Sec  Memory Limit: 128 MB

Description

周末同学们非常无聊,有人提议,咱们扔硬币玩吧,谁扔的硬币正面次数多谁胜利。大家纷纷觉得这个游戏非常符
合同学们的特色,但只是扔硬币实在是太单调了。同学们觉得要加强趣味性,所以要找一个同学扔很多很多次硬币
,其他同学记录下正反面情况。用H表示正面朝上,用T表示反面朝上,扔很多次硬币后,会得到一个硬币序列。比
如HTT表示第一次正面朝上,后两次反面朝上。但扔到什么时候停止呢?大家提议,选出n个同学,每个同学猜一个
长度为m的序列,当某一个同学猜的序列在硬币序列中出现时,就不再扔硬币了,并且这个同学胜利,为了保证只
有一个同学胜利,同学们猜的n个序列两两不同。很快,n个同学猜好序列,然后进入了紧张而又刺激的扔硬币环节
。你想知道,如果硬币正反面朝上的概率相同,每个同学胜利的概率是多少。

Input

第一行两个整数n,m。
接下里n行,每行一个长度为m的字符串,表示第i个同学猜的序列。
1<=n,m<=300

Output

输出n行,第i行表示第i个同学胜利的概率。
输出与标准输出的绝对误差不超过10^-6即视为正确。

Sample Input

3 3
THT
TTH
HTT

Sample Output

0.3333333333
0.2500000000
0.4166666667
 
题解:
生无可恋.jpg
在我做过的题里继小凸玩密室又一神题啊……我可能学了假的概率DP
上来我们可以发现这本是一道字符集为2的AC自动机的DP,迅速码好读入和处理fail指针。
然后我们继续套路,发现这个获胜概率在Trie图上互相牵制不能递推,所以我们要用高斯消元。
……高斯消元?
如果我们对于每个AC自动机的节点都这样做的话,仅时间复杂度就变成了O(3006),直接螺旋上天了。
那么我们只能不考虑每个节点,而是考虑每个串了。
设p[i]为第i个同学获胜的概率,也就是说第一个匹配到第i个串的概率。
首先我们可以列出第一个方程:Σp[i]=1.0
那么我们考虑,在一个不是单词节点的节点,假设已经经过的字符状态为S,
如果我们向后面添加m个字符,那么显然,由于我们是完全随机添加的,所以匹配到每个串i的概率都是一样的,我们设为H。
不难看出,如果我们匹配到i,那么这种添加方式可以包含所有匹配到i的情况。
但是这里H不一定等于p[i]:我们经过每个非单词节点的概率也不一定一样;
并且同时,由于我们之前已经匹配了状态为S的一些字符,我们可能在添加不到m个字符时就匹配到了某个串j(j可以等于i)
如果在某个节点加上字符串i的前k个字符后就已经到达了字符串j的终止节点,那么j的后k个字符必然等于i的前k个字符.
在匹配上j后,(虽然继续匹配是非法的,但是我们要减去这种非法状态,所以我们还要计算这种状态发生的概率.)
我们还要继续生成字符使得接下来m-k的字符等于串i的后m-k个字符,也就是说,p[i]应该在H的基础上减去p[j]*(1/2)m-k
这里的“前k个字符重叠”,我们可以利用KMP的失配指针来处理。
(其实这里的处理方法有很多,除了kmp,hash也可以,只要能找出两串的重叠部分即可)
那么最后,我们可以列出方程:p[i]=H-Σp[j]*(1/2)m-k,移项得p[i]+Σp[j]*(1/2)m-k-H=0
再加上一开始的方程Σp[i]=1.0,我们就可以对n+1个变量列出n+1个方程来解方程了!
代码见下:
 1 #include <cstdio>
 2 #include <cstring>
 3 #include <cmath>
 4 #include <algorithm>
 5 using namespace std;
 6 const int N=310;
 7 int n,l,cnt,fail[N<<1];
 8 char s[N][N],str[N<<1];
 9 double f[N][N];
10 inline void kmp()
11 {
12     register int i,j,len;
13     for(i=2,j=0,len=(l<<1);i<=len;++i)
14     {
15         while(j&&str[j+1]!=str[i])j=fail[j];
16         j=(str[j+1]==str[i])?j+1:j,fail[i]=j;
17     }
18 }
19 void Swap(int a,int b)
20     {for(register int i=1;i<=cnt+1;++i)swap(f[a][i],f[b][i]);}
21 void Execution(int a,int b,double t)
22     {for(register int i=1;i<=cnt+1;++i)f[a][i]+=f[b][i]*t;}
23 inline void gauss()
24 {
25     register int i,j,k;
26     for(i=1;i<=cnt;++i)
27     {
28         for(j=i+1;j<=cnt;++j)if(fabs(f[i][i])<fabs(f[j][i]))Swap(i,j);
29         for(j=1;j<=cnt;++j)if(j!=i)Execution(j,i,-f[j][i]/f[i][i]);
30     }
31     for(i=1;i<=cnt;++i)f[i][cnt+1]/=f[i][i];
32 }
33 int main()
34 {
35     scanf("%d%d",&n,&l);register int i,j,k;
36     for(i=1;i<=n;++i)scanf("%s",s[i]+1);
37     for(i=1;i<=n;++i)
38         for(j=1;j<=n;++j)
39         {
40             for(k=1;k<=l;++k)str[k]=s[i][k],str[l+k]=s[j][k];
41             for(kmp(),k=fail[l<<1];k;k=fail[k])
42                 if(k<l)f[i][j]+=pow(0.5,l-k);
43         }
44     for(i=1;i<=n;++i)f[i][i]+=1.0,f[i][n+1]-=1.0;//n+1代表未知变量H
45     for(i=1,f[n+1][n+2]=1.0;i<=n;++i)f[n+1][i]=1.0;
46     cnt=n+1;gauss();
47     for(i=1;i<=n;++i)printf("%.10lf\n",f[i][cnt+1]);
48 }
 
posted @ 2017-09-13 21:03  LadyLex  阅读(536)  评论(0编辑  收藏  举报