Manacher算法详解

Manacher 算法是求字符串最大回文子串最高效的算法,时间复杂度和空间复杂度都为O(n),相较于时间复杂度为O(n3)的暴力穷举和时间复杂度为O(n2)的动态规划算法具有明显的优势。算法的目的在于提高信息及资源的利用效率,减少不必要的计算。那么本文的主要内容主要分为两个部分,一是说明manacher算法的思想,二是给出相应java代码实现。抛砖引玉,如有错误请不吝赐教。

一 算法要点分析


求最大回文子串一般情况需要分奇数串和偶数串来进行分类讨论,而Manacher算法则通过一种处理方式将字符串都转化为奇数串来处理,即在首尾及每量个字符之间插入任意选定字符,比如在字符串 "asdsa" 中插 '#' 变为 "#a#s#d#s#a#" (其实插入任意字符都可以,因为插入后原字符仍与原字符进行比较,但是为了使初次接触此算法的同学便于理解,并且不去刻意做无用的“创新”,遂沿用‘#’,好像废话有点多)。

//字符串预处理
StringBuilder newStr = new StringBuilder();
newStr.append('#');
for (int i = 0; i < str.length(); i ++) {
     newStr.append(str.charAt(i));
     newStr.append('#');
    }

先求处理过的字符串每个位置上的最大回文字符串的回文半径r,这个半径 - 1 正好是原字符串以此位置为中心的最大回文字符串的长度。例如:

回文半径数组示图.png

所以问题将转化为求处理过的字符串每个位置上的最大回文字符串半径。

介绍算法前首先需要说明三个Manacher算法中会用到的三个概念。

  • 从左向右遍历字符串已找到的回文子串最右边的字符位置right。
  • 最右回文子串的中心位置id。
  • 记录每个位置为中心时最大回文字符串的半径长度数组rad[]。
对称位置回文半径两种情况示图.png

如上图所示,id为最右回文子串的中心,right为最右回文子串的右边界,当计算某位置id+k时,可以利用其关于id对称的位置id-k的最大回文半径,从而避免重复计算,提高效率。此时分两种情况:
1 当rad[id-k] >= rad[id] - k 时,如图橙色区间超出蓝色区间,id+k 的最大回文半径最小是rad[id]-k,如图中与红色区间关于id对称的绿色区间。

2 当rad[id-k] < rad[id] - k 时,如图橙色区间未超出蓝色区间,id+k 的最大回文半径最小是rad[id-k],如图中与红色区间关于id对称的绿色区间。
所以在任何情况下,rad[ id + k ] = min( rad [ id ] - k, rad[ id - k ] )。

二 完整函数代码


public static int getPalindromeLength(String str) {
        StringBuilder newStr = new StringBuilder();

        // 插入字符,统一奇偶性便于计算。
        newStr.append('#');
        for (int i = 0; i < str.length(); i++) {
            newStr.append(str.charAt(i));
            newStr.append('#');
        }

        // 用来记录每个位置的最大回文半径
        int[] rad = new int[newStr.length()];
        // 用来记录最右回文字符串的中心位置
        int id = -1;
        // 用来记录最右回文字符串的最右位置
        int right = -1;
        // 从左向右依次遍历
        for (int i = 0; i < newStr.length(); i++) {
            // 初始化每一位置最小半径至少为1
            int r = 1;
            // 从回文半径数获取计算过的回文半径,减少计算
            if (right >= i) {
                r = Math.min(rad[id] - i + id, rad[2 * id - i]);
                // 只有相等的情况才有可能找到最大的回文半径。
                if (rad[id] - i == rad[2 * id - i])
                    // 从上一步基础位置开会依次向两个方向比较,直到对称位置字符不同
                    while (i - r >= 0 && i + r < newStr.length()
                            && (newStr.charAt(i + r) == newStr.charAt(i - r))) {
                        r++;
                    }
            }

            // 当right相同是,取最左边的中心位置效率最高,所以不用更新。
            if (i + r - 1 > right) {
                right = i + r - 1;
                id = r;
            }
            // 更新回文半径数组
            rad[i] = r;

        }
        // 获取新字符串最大回文半径
        int maxLength = 1;
        for (int r : rad) {
            maxLength = r > maxLength ? r : maxLength;
        }
        // 原字符串最大回文长度为新字符串最大回文半径的长度 - 1
        return maxLength - 1;
    }

三 复杂度分析


* 空间复杂度 :

新字符串长度与原字符串长度为线性关系,回文半径数组长度为新字符串长度,所以空间复杂度为O(n)。

* 时间复杂度 :

时间复杂度分析示图.png

如上图所示,橙色区间为第id趟最大回文字符串,蓝色区间为第id+1趟最大回文字符串。

最坏情况下(全部字符相同)
从第三趟开始每次需要多比较两次(图中紫色区间),直到整个字符串的中心位置为止( ≈ 2 / n 趟),之后每个位置比较次数为0。

最好情况下(原字符串每个字符皆不相)
每次只需比较1到2次,直到遍历完整个字符串(n趟)。

综上,此算法的平均时间复杂度为O(n)。

四 总结


manather算法首先对原字符串做预处理,以便于统一计算,并充分利用了已经经过计算得出的信息,减少了计算量,在最好情况和最差情况都具有线性的复杂度。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,366评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,521评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,689评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,925评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,942评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,727评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,447评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,349评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,820评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,990评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,127评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,812评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,471评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,017评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,142评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,388评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,066评论 2 355