(Description)
给定一个字符串(s),求一个最短的字符串(t)满足,将(t)拼接多次后,可以得到(s)。拼接是指,可以将(t)放在当前串的任意位置,但要保证对应位置相同。(不太会说,看样例吧...)
(|s|leq5 imes10^5)。
(Solution)
首先(t)既是(s)的前缀也是(s)的后缀,即(s)的(border)、(border)的(border)...
考虑(KMP)建出(fail)树,那么(t)是(n)到根节点路径上的某个节点。考虑路径上怎样的节点是合法的。
(s)中能够放(t)的位置(i)满足,(t)是(s[1...i])的(border)或者(border)的(border)...即在(fail)树中(i)在(t)的子树内。对于所有(i),要满足相邻的(i)之间的最大间距不超过(|t|),这样(t)就是合法的。
注意到从根节点到(n)的路径中,要维护的这个最大间距是递减的(每次删掉若干棵子树中的(i))。那么删除一个位置时,用链表维护下相邻位置的最大间距就可以了。
还有一个更简单的DP方法,看洛谷题解吧,我还不是很理解就不写了。
//20760kb 320ms
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
typedef long long LL;
const int N=5e5+5;
int fail[N],H[N],nxt[N],way[N],L[N],R[N],Max;
char s[N];
inline void AE(int u,int v)
{
nxt[v]=H[u], H[u]=v;
}
void Delete(int x)
{
L[R[x]]=L[x], R[L[x]]=R[x], Max=std::max(Max,R[x]-L[x]);
for(int v=H[x]; v; v=nxt[v]) if(v!=way[x]) Delete(v);
}
int main()
{
scanf("%s",s+1); int n=strlen(s+1);
for(int i=2,j=0; i<=n; ++i)
{
while(j&&s[i]!=s[j+1]) j=fail[j];
fail[i]=s[i]==s[j+1]?++j:0;
}
for(int i=1; i<=n; ++i) L[i]=i-1, R[i]=i+1, AE(fail[i],i);
for(int x=n; x; x=fail[x]) way[fail[x]]=x;
int x=0;
for(Max=1; Max>x; x=way[x]) Delete(x);
printf("%d
",x);
return 0;
}