Annihilate(SA)

题目描述

黑暗之主的蜈蚣几乎可以毁灭一切,因此小正方形陷入了苦战……

小正方形现在需要减弱黑暗之主的攻击。

一个黑暗之主的攻击可以用一个仅有小写字母的字符串表示。

现在黑暗之主向小正方形发动了若干攻击,对于两个攻击,小正方形能选出它们最长的公共子串,并把这一段消除。

现在小正方形想要知道,对于任意两个黑暗之主的攻击,它们的最长公共子串长度是多少,你能帮帮它吗?

题解

我是个**,字符集会随着SA往后进行越来越大,数组开小调了俩小时。。

用SAM是板子,但它卡空间,所以把它接起来跑一边SA。

我们按照字典序扫描所有后缀,维护一个数组,表示在当前位置到之前x个串的LCS,每次O(n)更新一遍。

代码

#include<iostream>
#include<cstdio>
#define N 2100002
using namespace std;
char s[N];
int rnk[N],y[N],sa[N],n,x,ji[N],m,tong[N],pre[52],height[N],ans[52][52],tot;
inline void init(int x){
    char c=getchar();
    while(!isalpha(c))c=getchar();
    while(isalpha(c))s[++n]=c,ji[n]=x,c=getchar();
} 
inline void qsort(){
    for(int i=0;i<=m;++i)tong[i]=0;
    for(int i=1;i<=n;++i)tong[rnk[i]]++;
    for(int i=1;i<=m;++i)tong[i]+=tong[i-1];
    for(int i=n;i>=1;--i)sa[tong[rnk[y[i]]]--]=y[i];
}
inline void SA(){
    for(int i=1;i<=n;++i)rnk[i]=s[i],y[i]=i;
    m=200;qsort();
    for(int w=1,p;p<n;m=p,w<<=1){
        p=0;
        for(int i=n-w+1;i<=n;++i)y[++p]=i;
        for(int i=1;i<=n;++i)if(sa[i]>w)y[++p]=sa[i]-w;
        qsort();
        swap(rnk,y);rnk[sa[1]]=p=1;
        for(int i=2;i<=n;++i)rnk[sa[i]]=((y[sa[i-1]]==y[sa[i]])&&(y[sa[i-1]+w]==y[sa[i]+w]))?p:++p;
    }
    for(int i=1;i<=n;++i){
        if(rnk[i]==1)continue;
        int j=max(0,height[rnk[i-1]]-1);
        while(s[i+j]==s[sa[rnk[i]-1]+j])j++;
        height[rnk[i]]=j;
    }
    for(int i=1;i<=n;++i){
       for(int j=1;j<=tot;++j)pre[j]=min(pre[j],height[i]);
       pre[ji[sa[i-1]]]=max(pre[ji[sa[i-1]]],height[i]);
       for(int j=1;j<=tot;++j)ans[ji[sa[i]]][j]=ans[j][ji[sa[i]]]=max(ans[j][ji[sa[i]]],pre[j]);
    }
}
int main(){
    scanf("%d",&tot);
    for(int i=1;i<=tot;++i){init(i);s[++n]=i;}
    SA();
    for(int i=1;i<=tot;++i){
        for(int j=1;j<=tot;++j)if(i!=j)printf("%d ",ans[i][j]);
        puts("");
    }
    return 0;
}
 
原文地址:https://www.cnblogs.com/ZH-comld/p/10189133.html