题解:
尝试学习sam第三次 ,这次我终于好了。
再次写这道题是跑slink的拓扑序得到的。这里说一下,如果我们从S状态到T状态跑,我们找的是x这个子串,在跑这个状态的过程中相当于x就是这些后缀的前缀,而我们每按照slink这个树跑一个状态我们后缀就会多一个数,子串出现次数也就会++,比如我们d,cd,bcd,abcd中d都是他们的子串所以我们每个子串出现的次数就是slink树中子树的大小。
#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e6+10;
char s[maxn];
vector<int> g[maxn*2];
int f[maxn*2];
long long res=0;
struct SuffixAutoMation
{
int last=1,cnt=1;
int trans[maxn<<1][26],slink[maxn<<1],l[maxn<<1];
int t[maxn<<1],a[maxn<<1];
int num[maxn<<1];
inline void add(int x)
{
int p = last,np = ++cnt;last = np;l[np] = l[p]+1; f[np]=1;
for(;p && !trans[p][x];p=slink[p]) trans[p][x] = np;
if(!p) slink[np] = 1;
else{
int q = trans[p][x];
if(l[p]+1 == l[q]) slink[np] = q;
else
{
int nq = ++cnt;l[nq] = l[p]+1;
memcpy(trans[nq],trans[q],sizeof(trans[q]));
slink[nq] = slink[q];
slink[q] = slink[np] = nq;
for(;trans[p][x] == q; p = slink[p]) trans[p][x] = nq;
}
}
num[np] = 1;
}
void build()
{
scanf("%s",s+1);
int len = strlen(s+1);
last = cnt = 1;
for(int i=1;i<=len;i++) add(s[i]-'a');
}
void init(){
for(int i=2;i<=cnt;i++) g[slink[i]].push_back(i);
}
void dfs(int x){
for(int i=0;i<g[x].size();i++){
int j=g[x][i];
dfs(j);
f[x]+=f[j];
if(f[x]>1) res=max(res,1LL*l[x]*f[x]);
}
return ;
}
}sam;
signed main()
{
sam.build();
sam.init();
sam.dfs(1);
cout<<res<<endl;
}