UVALive - 3490 Generator (AC自动机+高斯消元dp)

初始有一个空串s,从前n个大写字母中不断随机取出一个字母添加到s的结尾,出现模式串t时停止,求停止时s的长度期望。

这道题解法不唯一,比较无脑的方法是对模式串t建一个单串AC自动机,设u为自动机上的一个结点,dp[u]为从该结点出发走到终结状态时的期望步数,则dp[u]=∑(1+dp[v])/n,v为u的后继状态。特别地,终结状态的dp值为0。

这样一来,就可以列出线性方程组进行高斯消元了。由于答案非常大,用double会损失精度,所以改成longlong。

由于有除法的存在,为了防止出现除不开的现象,我写了个大模数+逆元的版本,还需要用到按位乘。(闲的蛋疼系列)

  1 #include<bits/stdc++.h>
  2 using namespace std;
  3 typedef long long ll;
  4 
  5 const ll mod=(1ll<<61)-1;
  6 int m,ka;
  7 char s[15];
  8 ll Mul(ll x,ll p) {
  9     if(p<0)p+=mod;
 10     ll ret=0;
 11     for(; p; x=(x+x)%mod,p>>=1)if(p&1)ret=(ret+x)%mod;
 12     return ret;
 13 }
 14 ll Pow(ll x,ll p) {
 15     ll ret=1;
 16     for(; p; x=Mul(x,x)%mod,p>>=1)if(p&1)ret=Mul(ret,x)%mod;
 17     return ret;
 18 }
 19 ll inv(ll x) {return Pow(x,mod-2);}
 20 
 21 struct Mat {
 22     static const int N=15;
 23     int n;
 24     ll a[N][N];
 25     ll* operator[](int x) {return a[x];}
 26     void gauss() {
 27         for(int i=0; i<n; ++i) {
 28             int r=i;
 29             if(!a[r][i]) {
 30                 for(int k=i+1; k<n; ++k)if(a[k][i]) {r=k; break;}
 31             }
 32             if(r!=i)for(int j=i; j<=n; ++j)swap(a[i][j],a[r][j]);
 33             for(int k=i+1; k<n; ++k) {
 34                 ll tmp=Mul(a[k][i],inv(a[i][i]));
 35                 for(int j=i; j<=n; ++j)a[k][j]=(a[k][j]-Mul(a[i][j],tmp))%mod;
 36             }
 37         }
 38         for(int i=n-1; i>=0; --i) {
 39             for(int j=i+1; j<n; ++j)a[i][n]=(a[i][n]-Mul(a[j][n],a[i][j]))%mod;
 40             a[i][n]=Mul(a[i][n],inv(a[i][i]));
 41         }
 42     }
 43 } G;
 44 
 45 struct AC {
 46     static const int N=100+10,M=26;
 47     int go[N][M],pre[N],tot,end[N];
 48     int idx(char ch) {return ch-'A';}
 49     void init() {tot=0; newnode();}
 50     int newnode() {
 51         int u=tot++;
 52         memset(go[u],0,sizeof go[u]);
 53         end[u]=pre[u]=0;
 54         return u;
 55     }
 56     void ins(char* s) {
 57         int u=0,n=strlen(s);
 58         for(int i=0; i<n; u=go[u][idx(s[i])],++i)
 59             if(!go[u][idx(s[i])])go[u][idx(s[i])]=newnode();
 60         end[u]=1;
 61     }
 62     void build() {
 63         queue<int> q;
 64         for(int i=0; i<M; ++i)if(go[0][i])q.push(go[0][i]);
 65         while(!q.empty()) {
 66             int u=q.front();
 67             q.pop();
 68             for(int i=0; i<M; ++i) {
 69                 if(go[u][i]) {
 70                     pre[go[u][i]]=go[pre[u]][i];
 71                     q.push(go[u][i]);
 72                 } else go[u][i]=go[pre[u]][i];
 73             }
 74         }
 75     }
 76     void gettransmat(Mat& G) {
 77         G.n=tot;
 78         int n=tot;
 79         for(int i=0; i<n; ++i)
 80             for(int j=0; j<=n; ++j)
 81                 G[i][j]=j==i?1:0;
 82         for(int i=0; i<tot; ++i)if(!end[i]) {
 83                 for(int j=0; j<m; ++j)G[i][go[i][j]]=(G[i][go[i][j]]-inv(m))%mod;
 84                 G[i][n]=(G[i][n]+1)%mod;
 85             }
 86     }
 87 } ac;
 88 
 89 int main() {
 90     int T;
 91     scanf("%d",&T);
 92     while(T--) {
 93         if(ka)puts("");
 94         printf("Case %d:\n",++ka);
 95         ac.init();
 96         scanf("%d%s",&m,s);
 97         ac.ins(s);
 98         ac.build();
 99         ac.gettransmat(G);
100         G.gauss();
101         printf("%lld\n",(G[0][G.n]+mod)%mod);
102     }
103     return 0;
104 }

 

posted @ 2019-02-01 11:47  jrltx  阅读(271)  评论(0编辑  收藏  举报