LG P4548 [CTSC2006]歌唱王国

Description

在歌唱王国,所有人的名字都是一个非空的仅包含整数 $1\sim n$ 的字符串。

王国里生活着一大群咕噜兵,他们靠不停地歌唱首领——牛人酋长们的名字来获取力量。咕噜兵每一次歌唱过程是这样的:首先,他从整数生成器那儿获得一个数字,然后花一个时间单位将此数字唱出来,如果他发现某个牛人酋长的名字已经被歌唱出来(即此名字是歌唱序列的一个连续子串),那么这次歌唱过程就立即结束。

相关名词定义:

- 歌唱序列:如果某人歌唱了 $x$ 个数字,第 $i$ 次歌唱的数字为 $a_i$,那么歌唱序列 $=(a_1,a_2,\cdots,a_x)$。
- 整数生成器:歌唱王国的神物,它有一个按钮,如果你按一下按钮,将从 $1\sim n$ 数字中等概率的随机返回一个整数。
- 歌唱时间:在一次歌唱过程中花费的时间。

歌唱时间是随机的,无法预料;不过歌唱时间的期望值是固定的,此期望值即平均来说歌唱时间有多长,亦可称作平均歌唱时间。

王国里的人非常喜欢歌唱,他们希望歌唱的时间越长越好,所以他们决定罢免一些牛人酋长,使得平均歌唱时间变长。但是他们不能罢免掉所有的牛人酋长,否则他们每次歌唱都无法停止,无法获取力量;于是他们决定只保留一个牛人酋长而罢免其余的牛人酋长。

你的任务是:对于给定的 $n$、牛人酋长的个数 $t$ 以及每一个牛人酋长的名字,告诉王国里的人们,对于 $1\leq i\leq t$,如果保留第 $i$ 个牛人酋长,罢免掉其余的,那么平均歌唱时间将是多少。

提示:此数为一个非负整数!

输出要求:由于这个数字太大,所以你只需输出这个数的末 $4$ 位数字。如果不足 $4$ 位,则前面补 $0$

Solution

设$f_x$为长度为$x$时成功匹配的概率,$g_x$为长度为$x$时尚未成功匹配的概率

写出$f$的概率生成函数

$$F(x)=\sum_{i=0}^{+ \infty} P_i x_i$$

同样的写出$g$的

发现$F(1)=1,F'(1)=E(x),F(x)+G(x)=G(x)x+1$

上式的意义类似于DP转移

还有一关系式,设$m$为字符集大小,$l$为字符串长度,$a$是该位是否为border的布尔数组

$$\left ( \frac{1}{m} \right ) ^lg_x =\sum _{i=1}^l a_i\left ( \frac{1}{m} \right )^{l-i}f_{x+i} $$

考虑在没有完成匹配的字符串后面接一个随机出一个原串的概率,右半部分是有可能产生原串的border导致提前匹配

写成生成函数的形式

$$\left ( \frac{x}{m} \right ) ^lG(x) =F(x)\sum _{i=1}^l a_i \left ( \frac{x}{m} \right ) ^{l-i}$$

对一式求导得

$$F'(x)+G'(x)=xG'(x)+G(x)$$

移项得

$$F'(x)=(x-1)G'(x)+G(x)$$

将$1$代入$x$

$$F'(1)=G(1)=E(x)$$

这非常屌,再搞另一个式子,已经知道要算$G(1)$,所以也代入$1$

$$G(1)=\sum_{i=1}^l a_im^i$$

KMP或者hash胡搞,$O(l)$求解

#include<iostream>
#include<cstdio>
using namespace std;
int n,t,m;
bool a[100005];
unsigned long long b[100005]={1},str[100005],has[100005],pn[100005]={1},ans;
const unsigned long long mod=10000,base=1e9+7;
inline int read()
{
    int f=1,w=0;
    char ch=0;
    while(ch<'0'||ch>'9'){if(ch=='-') f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9') w=(w<<1)+(w<<3)+ch-'0',ch=getchar();
    return f*w;
}
int main()
{
    n=read(),t=read();
    for(int i=1;i<=100000;i++) b[i]=b[i-1]*base;
    for(int i=1;i<=100000;i++) pn[i]=pn[i-1]*n%mod;
    for(;t;t--)
    {
        m=read(),ans=0;
        for(int i=1;i<=m;i++) str[i]=read(),has[i]=has[i-1]*base+str[i];
        for(int i=1;i<=m;i++) a[i]=(has[i]==has[m]-has[m-i]*b[i]);
        for(int i=1;i<=m;i++) (ans+=a[i]*pn[i]%mod)%=mod;
        if(ans<1000) printf("0");
        if(ans<100) printf("0");
        if(ans<10) printf("0");
        printf("%d\n",ans);
    }
    return 0;
}
[CTSC2006]歌唱王国

 

posted @ 2020-12-06 09:44  QDK_Storm  阅读(133)  评论(0编辑  收藏  举报