manacher算法

manacher算法是在O(n)的复杂度内求回文串长度的算法。

算法过程如下。

先在所有字符之间加上一种没有意义的字符。

比如“#”,“|”等。来去除偶数回文和奇数回文的区别。

再在第0位加上“~”,这样就可以保证不会出范围。

其中rb表示当前mid的回文串右边界。

枚举中间点 i 如果 i 在右边界之前。

p[i],也就是i时的回文半径。 j 为 i 关于mid的对称点。

关于mid的更新在下面再说。

p[ i ] = min( p[ j ] , rb-i );

这句话尝试用j的回文长度来求i的回文长度。

因为关于mid对称,所以p[i]可以等于p[j]。

如果i在右边界之后,则p[i]=1;

那么就可以继续向后查。

while(da[i-p[i]]==da[i+p[i]])

  p[i]++;

将半径扩大。

这时候如果p[i]+i大于rb,也就是现在的范围超出了原来的范围。

那么可以更新rb=p[i]+i;  那么 mid=i; 

总体算法结束。

那么最后的答案,最长回文长度就是最长半径-1。(为什么减一?规律哈哈哈哈)

模板:https://www.luogu.org/problemnew/show/P3805

代码如下:

#include<bits/stdc++.h>
using namespace std;
const int maxn=11000001;
char da[maxn<<1];
int p[maxn<<1],cnt=1;
int rb,maxx,mid;
int main()
{
    char c=getchar();
    da[0]='~';
    da[1]='#';
    while(c>'z'||c<'a')
        c=getchar();
    while(c<='z'&&c>='a'){
        da[++cnt]=c;
        da[++cnt]='#';
        c=getchar();
    }
    for(int i=1;i<=cnt;i++){
        if(i<rb)
            p[i]=min(p[(mid<<1)-i],rb-i);
        else p[i]=1;
        while(da[i-p[i]]==da[i+p[i]])
            p[i]++;
        if(p[i]+i>rb)
            rb=p[i]+i,mid=i;
        maxx=max(p[i],maxx);
    }
    printf("%d\n",maxx-1);
    // while(1);
    // system("pasue");
    return 0;
}

 

posted @ 2019-07-10 21:21  ChrisKKK  阅读(222)  评论(1编辑  收藏  举报