把三个串加上ASCII大于z的分隔符连起来,然后求SA

显然每个相同子串都是一个后缀的前缀,所以枚举s1的每个后缀的最长和s2相同的前缀串(直接在排序后的数组里挨个找,最近的两个分别属于s1和s2的后缀的height一定是最长符合要求的前缀),然后判断一下这个子串里最早出现的和s3相同的子串的位置,这里先预处理出每个等于s3的子串的位置然后二分找(没有就直接和height取max),然后答案就是从这个后缀开头到和s3相同子串结尾的前一个位置的长度

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=200005;
int l1,l2,l3,n,wa[N],wb[N],wv[N],wsu[N],sa[N],rk[N],he[N],b[N],st[20][N],ans,bl[N],a[N],tot;
char s[N],c1[N],c2[N];
bool cmp(int r[],int a,int b,int l)
{
return r[a]==r[b]&&r[a+l]==r[b+l];
}
void saa(char r[],int n,int m)
{
int *x=wa,*y=wb;
for(int i=0;i<=m;i++)
wsu[i]=0;
for(int i=1;i<=n;i++)
wsu[x[i]=r[i]]++;
for(int i=1;i<=m;i++)
wsu[i]+=wsu[i-1];
for(int i=n;i>=1;i--)
sa[wsu[x[i]]--]=i;
for(int j=1,p=1;j<=n&&p<n;j<<=1,m=p)
{
p=0;
for(int i=n-j+1;i<=n;i++)
y[++p]=i;
for(int i=1;i<=n;i++)
if(sa[i]>j)
y[++p]=sa[i]-j;
for(int i=1;i<=n;i++)
wv[i]=x[y[i]];
for(int i=0;i<=m;i++)
wsu[i]=0;
for(int i=1;i<=n;i++)
wsu[wv[i]]++;
for(int i=1;i<=m;i++)
wsu[i]+=wsu[i-1];
for(int i=n;i>=1;i--)
sa[wsu[wv[i]]--]=y[i];
swap(x,y);
p=1;
x[sa[1]]=1;
for(int i=2;i<=n;i++)
x[sa[i]]=cmp(y,sa[i-1],sa[i],j)?p:++p;
}
for(int i=1;i<=n;i++)
rk[sa[i]]=i;
for(int i=1,j,k=0;i<=n;he[rk[i++]]=k)
for(k?k--:0,j=sa[rk[i]-1];r[i+k]==r[j+k];k++);
}
int ques(int l,int r)
{
l++;
int k=b[r-l+1];
return min(st[k][l],st[k][r-(1<<k)+1]);
}
int ef(int x)
{
int l=1,r=tot,ans=0;
while(l<=r)
{
int mid=(l+r)>>1;
if(a[mid]>=x)
r=mid-1,ans=mid;
else
l=mid+1;
}
return ans;
}
int main()
{
scanf("%s%s%s",s+1,c1+1,c2+1);
n=l1=strlen(s+1),l2=strlen(c1+1),l3=strlen(c2+1);
s[++n]='z'+1;
for(int i=1;i<=l2;i++)
s[++n]=c1[i];
s[++n]='z'+2;
for(int i=1;i<=l3;i++)
s[++n]=c2[i];
// for(int i=1;i<=n;i++)
// cerr<<s[i];cerr<<endl;
saa(s,n,200);
// for(int i=1;i<=n;i++)
// cerr<<sa[i]<<" "<<he[i]<<endl;cerr<<endl;
b[0]=-1;
for(int i=1;i<=n;i++)
st[0][i]=he[i],b[i]=b[i>>1]+1;
for(int i=1;i<=17;i++)
for(int j=1;j+(1<<i)-1<=n;j++)
st[i][j]=min(st[i-1][j],st[i-1][j+(1<<(i-1))]);
for(int i=1;i<=n;i++)
{
if(sa[i]<=l1)
bl[i]=1;
else if(sa[i]>l1+1&&sa[i]<=l1+1+l2)
bl[i]=2;
else if(sa[i]>l1+l2+2)
bl[i]=3;
}
a[++tot]=l1+l2+3;
for(int i=rk[l1+l2+3]+1;i<=n;i++)
{
if(he[i]>=l3)
a[++tot]=sa[i];
else
break;
}
for(int i=rk[l1+l2+3]-1;i>=1;i--)
{
if(he[i+1]>=l3)
a[++tot]=sa[i];
else
break;
}
sort(a+1,a+1+tot);
// for(int i=1;i<=tot;i++)
// cerr<<a[i]<<" ";cerr<<endl;
int la;
for(la=1;la<=n;la++)
if(bl[la]==1||bl[la]==2)
break;
for(int i=la+1;i<=n;i++)
if(bl[i]==1||bl[i]==2)
{
if(bl[i]!=bl[la])
{
int p=ef(sa[i]),len=ques(la,i);
if(p&&a[p]+l3-1<=sa[i]+len-1)
ans=max(ans,a[p]-sa[i]+l3-1);//,cerr<<sa[i]<<" "<<sa[i-1]<<" "<<len<<" "<<a[p]<<" "<<a[p]-sa[i]+l3-1<<endl;
else
ans=max(ans,len);
}
la=i;
}
printf("%d\n",ans);
return 0;
}

最新文章

  1. [deviceone开发]-do_Http组件示例
  2. 使用 python 获取 Linux 的 IP 信息(通过 ifconfig 命令)
  3. Orchard源码分析(1):Orchard架构
  4. POJ 3373 Changing Digits(DP)
  5. html 字体加粗
  6. timestamp 与 rowversion
  7. 1497: [NOI2006]最大获利 - BZOJ
  8. 【原创】LoadRunner Java Vuser开发环境配置指南
  9. Spring 3.0 + Atomikos构建jta分布式事务
  10. 如果用float实现居中
  11. bigdata_hive_Issue of Vectorization on Parquet table
  12. 简单工厂设计模式(Simple Factory Design Pattern)
  13. 从并发处理谈PHP进程间通信(一)外部介质
  14. Python 并发编程(一)之线程
  15. Java基础点滴
  16. Gradle-----搭建简单的Gradle项目
  17. 你可以这么理解五种I/O模型
  18. linux提取第一列且删除第一行(awk函数)
  19. 如何让Excel单元格中的名字分散对齐
  20. 03-封装BeanUtil工具类(javabean转map和map转javabean对象)

热门文章

  1. stl之multiset容器的应用
  2. HTML--比较实用的小例子
  3. IE8.0登录Oracle EBS后报Oracle error 1403错
  4. SPOJ VLATTICE Visible Lattice Points (莫比乌斯反演基础题)
  5. 项目Alpha冲刺(团队10/10)
  6. A new session could not be created. (Original error: Requested a new session but one was in progress) )错误解决办法
  7. notepad++的f90配置文件
  8. HDU 6125 Free from square 状态压缩DP + 分组背包
  9. split+ Pattern切割字符串
  10. 如何使用Visual Studio构建libiconv