求最大的公共前缀;
用后缀数组做;
其实暴力也可以过;
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 2009
using namespace std; int s[maxn];
int sa[maxn],t[maxn],t2[maxn],c[maxn],n; void build_sa(int m)
{
int *x=t,*y=t2;
for(int i=; i<m; i++)c[i]=;
for(int i=; i<n; i++)c[x[i]=s[i]]++;
for(int i=; i<m; i++)c[i]+=c[i-];
for(int i=n-; i>=; i--)sa[--c[x[i]]]=i;
for(int k=; k<=n; k<<=)
{
int p=;
for(int i=n-k; i<n; i++)y[p++]=i;
for(int i=; i<n; i++)if(sa[i]>=k)y[p++]=sa[i]-k;
for(int i=; i<m; i++)c[i]=;
for(int i=; i<n; i++)c[x[y[i]]]++;
for(int i=; i<m; i++)c[i]+=c[i-];
for(int i=n-; i>=; i--)sa[--c[x[y[i]]]]=y[i];
swap(x,y);
p=;
x[sa[]]=;
for(int i=; i<n; i++)
x[sa[i]]=y[sa[i-]]==y[sa[i]]&&y[sa[i-]+k]==y[sa[i]+k]?p-:p++;
if(p>=n)break;
m=p;
}
} int rank[maxn],height[maxn];
void getheight()
{
int k=;
for(int i=; i<n; i++)rank[sa[i]]=i;
for(int i=; i<n; i++)
{
if(k)k--;
int j=sa[rank[i]-];
while(s[i+k]==s[j+k])k++;
height[rank[i]]=k;
}
} bool check(int mid)
{
int mi=,ma=-;
for(int i=; i<n; i++)
{
if(height[i]>=mid)
{
mi=min(mi,sa[i]);
ma=max(ma,sa[i]);
}
else
{
if(ma-mi>=mid)return ;
mi=ma=sa[i];
}
}
if(ma-mi>=mid)return ;
return ;
} bool isbad[];
char bad[],good[];
int main()
{
int tt,r,ca=;
scanf("%d",&tt);
getchar();
while(tt--)
{
memset(isbad,,sizeof isbad);
gets(good);
gets(bad);
int l=strlen(bad);
for(int i=; bad[i]!='\0'; i++)
isbad[bad[i]]=;
n=strlen(good);
while(n>&&good[n-]==' ')
{
good[n-]='\0';
--n;
}
int xx='z';
for(int i=; good[i]!='\0'; i++)
{
if(isbad[good[i]])s[i]=xx++;
else s[i]=good[i];
}
s[n]=;
if (n<=)
{
printf("Case #%d: 0\n",ca++);
continue;
}
n++;
build_sa(xx+);
getheight();
l=,r=n>>;
while(l<=r)
{
int mid=(l+r)>>;
if(check(mid))l=mid+;
else r=mid-;
}
printf("Case #%d: %d\n",ca++,r);
}
return ;
}