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'的最长回文串前一字符和下一字符为xy,x不等于y,所以i位置的最长回文串的上一和下一字符为mn,m和n肯定不是相等的,因为m等于y,n等于x(因为这两个子串都被包含于[L,R]中)

    • i位置的对称i'的最长回文串跨过了最大左边界L,这时i位置的最长回文半径就是右边界Ri的长度。因为假设L的上一字符为x,R的下一字符为y,x和y肯定是不相等的,因为x和y都在最大的回文串[L,R]外,说明i为位置的最大回文串半径就是R-i的长度

    • i位置的对称i'的最长回文串恰好在最大左边界L,这是i位置的最长回文串半径不知道,只能知道最小回文串是右边界Ri的长度,此时仍需从下标i开始向左右两边扩散求最长回文串。

下面是java版本:

Last updated

Was this helpful?