BZOJ2915 : [Poi1997] gen

设f[i][j]表示串ij可以由哪些字母成长过来,用二进制压位表示。

设g[i][j]表示给定串中[i,j]这个区间一开始可以由哪些字母成长多来,用二进制压位表示。

设h[i]表示给定串前i位最少需要几个字母,h[i]=min(h[j]+1),j<i且[j+1,i]可以由S成长过来。

时间复杂度$O(n^3)$。

 

#include<cstdio>
#include<cstring>
#define N 105
int T,n,i,j,k,f[26][26],g[N][N],h[N];char a[N];
inline int cal(int U,int y){
  int t=0;
  for(;U;U-=U&-U)for(int i=__builtin_ctz(U&-U),V=y;V;V-=V&-V)t|=f[i][__builtin_ctz(V&-V)];
  return t;
}
int main(){
  for(scanf("%d",&n);n--;f[a[1]-'A'][a[2]-'A']|=1<<(a[0]-'A'))scanf("%s",a);
  for(scanf("%d",&T);T--;){
    for(scanf("%s",a+1),n=std::strlen(a+1),i=1;i<=n;i++)a[i]-='A';
    for(i=n;i;i--)for(g[i][i]=1<<a[i],j=i+1;j<=n;j++)for(g[i][j]=0,k=i;k<j;k++)g[i][j]|=cal(g[i][k],g[k+1][j]);
    for(i=1;i<=n;i++)for(h[i]=N,j=0;j<i;j++)if((g[j+1][i]&262144)&&h[j]+1<h[i])h[i]=h[j]+1;
    if(h[n]<N)printf("%d\n",h[n]);else puts("NIE");
  }
  return 0;
}

  

posted @ 2015-08-14 20:50  Claris  阅读(591)  评论(0编辑  收藏  举报