Manacher
title: Manacher mathjax: true data: 2020-03-24 19:20:31 updated: tags: -最长回文字串 categories:
-算法
manacher算法是用来求最长回文字串的,最长回文字串有一种经典求法:就是在源字符串中的每个字符前添加一个额外字符(这个额外字符并要求一定是源字符串中没有出现过的字符),添加完后,依次求每一个位置上的最大回文字串,然后除以2,就是源字符串对应位置的最长回文字串。例如:122131221
改为#1#2#2###1#2#2#1#
.。 那么如果不添加求呢?不添加的话,会错过长度为偶数的最长回文字串。
manacher算法中有以下几个概念(适用范围:添加额外字符后字符串),以#1#2#2#1#3#1#2#2#1#
为例:
回文直径:一个位置上的最大回文子串
回文半径:一个位置上的最大回文字串的一半,要包含中点,比如字串
#1#2#2#1#
,左2
位置的回文半径为2,包括2#
,完整回文串为#2#
,回文直径为3最大右边界:一个位置上的最大回文字串最右边界位置:比如
#1#2#2#1#
,左2
的最大右边界为4,如果有新位置产生了新的最大右边界,那么需要更新取得最大右边界时的中心点:仍拿上述字串为例:
#1#2#2#1#
,左2
的中心点位置为3。
manacher算法求最大回文字串有两种大分支,第二个大分支有三种小情况:
第一种情况:当前i位置大于最大右边界R,这时就硬求最长回文字串
第二种情况:当前i位置小于最大有边界R,这是有三种情况:
i
位置的对称i'
的最长回文字串完全在最大左边界L
的右侧,即完全被包含,这时i'
的最长回文串半径就是i
位置的最长回文串半径,假设i'
的最长回文串前一字符和下一字符为x
和y
,x不等于y,所以i
位置的最长回文串的上一和下一字符为m
和n
,m和n肯定不是相等的,因为m等于y,n等于x(因为这两个子串都被包含于[L,R]中)i
位置的对称i'
的最长回文串跨过了最大左边界L
,这时i
位置的最长回文串长度就是右边界R
到i
的长度。因为假设L的上一字符为x
,R的下一字符为y
,x和y肯定不是相等的,因为x和y都在最大的回文串[L,R]外,说明i为位置的最大回文串半径就是R-i
的长度i
位置的对称i'
的最长回文串恰好在最大左边界L
,这是i
位置的最长回文串半径不知道,只能知道最小回文串是右边界R
到i
的长度,此时仍需向左右两边扩散求最长回文串。
Last updated
Was this helpful?