最长回文如何解决?

新年之际,给大家带来了一道题目并且解决方法。希望大家喜欢

题目:

给定字符串,找到它的最长回文子串,都有哪些思路呢?例如"adaiziguizhongrenenrgnohziugiziadb",回文字串很多了,但最长的是"daiziguizhongrenenrgnohziugiziad"。

下面简单的介绍5种解决方法:

方法一

暴力法,外面的两层循环找到所有子串,第三层循环判断子串是否是回文。

function makeOdd(str){  
    var len = str.length;  
    if(len % 2 === 1){  
        return str;  
    }  
    var newStr = '#';  
    for(i = 0;i<len;i++){  
        newStr += str[i]+'#';  
    }  
    return newStr;  
}  
function judge(str){  
    (str.length%2 === 0) && (str = makeOdd(str));  
    var len = str.length,  
        half = Math.floor(len/2),  
        last = len-half;  
    var i = 0;  
    while(i<=last){  
        if(str[half-i] !== str[half+i]){  
            return false;  
        }  
        i++;  
    }  
    return true;  
}  
function getAllSubs(str){  
    var len = str.length,  
        res = [];  
    for(var i = 0;i<len;i++){  
        for(var j = i;j<len;j++){  
            var sub = str.substring(i,j+1);  
            console.error(sub);  
            if(sub && judge(sub)){  
                res[res.length] = sub;  
            }  
        }  
    }  
    return res;  
}  
console.log(getAllSubs('aaaabaac'));  

方法二:

动态规划的方法。开辟一个P[i][j]用来表示str[i..j]是否为回文,P[i][j]的状态转移方程如下:
当i==j时,P[i][j]=true
当i+1==j时,P[i][j]=str[i]==str[j]
其他,P[i][j]=P[i+1][j-1]&&(str[i]==str[j])

// 获取所有的子串回文情况  
    function getP2(str){  
        var len = str.length,  
            gap = '_';  
        var p = {};  
        var i,j,L;  
        // 只有一个字符的情况是回文  
        for( i =0;i<len;i++){  
            p[i+gap+i] = true;  
        }  
        // L是i和j之间的间隔数(因为间隔数从小到大渐增,所以大的间隔数总能包含小的间隔数)  
        for(L=2;L<=len;L++){  
            // 从0开始  
            for(i=0;i<=len-L;i++){  
                j = i+L-1;  
                if(L === 2){  
                    p[i+gap+j] = (str[i] === str[j]);  
                }else{  
                    p[i+gap+j] = (str[i]===str[j])&&p[i+1+gap+(j-1)];  
                }  
            }  
        }  
        re

方法三:

可以从上面那个方法的状态转移方程获得启发,对于每一个回文子串可以先确定一个中心,然后向两边扩展,这样可以在时间复杂度O(n^2),空间复杂度O(1)的情况下完成,需要注意的是,长度为奇数和偶数的中心的情况是不同的。

function getP3(str){  
    var maxLength = 1,  
        start = 0,  
        len = str.length,  
        low,high;  
    for(var i=1;i<len;++i){  
        low = i-1;  
        high = i;  
        while(low>=0 && high < len && str[low] === str[high]){  
            if(high-low+1>maxLength){  
                start = low;  
                maxLength = high-low+1;  
            }  
            --low;  
            ++high;  
        }  
    }  
    low = i-1;  
    high = i+1;  
    while(low>=0 && high < len && str[low] === str[high]){  
        if(high-low+1>maxLength){  
            start = low;  
            maxLength = high-low+1;  
        }  
        --low;  
        ++high;  
    }  
    return maxLength;  
}  

方法四:

第四个方法采用后缀数组,将最长回文子串的问题转化为最长公共前缀的问题。具体的做法就是:将整个字符串翻转之后,拼接到原字符串后,注意用特殊字符分开,这样问题就变成了新的字符串的某两个后缀的最长公共前缀的问题了。这个方法比较强大,很多字符串的问题都能够巧妙的解决。不过实现起来也相对比较难,好的实现和差的实现时间复杂度相差很大。

后缀数组是一种对字符串处理的比较有力的实现方案,可以用于求解最大公共子串、最长回文串等问题。但是其实现思路还是比较难以理解的。基本的思想就是获取字符串的后缀字符串,然后获得后缀字符串的最长公共前缀。其中牵扯到一系列的表示(如后缀数组、排序数组等),对于如何求得这些表示又有不同的对应方案。具体的论文可参考:后缀数组——处理字符串的有力工具

方法五:

第五个方法叫做Manacher算法,是一种线性时间的方法,非常巧妙。引入一个技巧,可以使得奇数和偶数的情况统一处理。具体做法如下:abba转换为#a#b#b#a#,也就是在每一个字符两边都加上一个特殊字符,这样,不管是奇数长度的字符串还是偶数长度的字符串就都能转化为奇数长度的字符串了。
然后创建一个新的P[i]表示,以第i个字符为中心的回文字串的半径。例如上面的例子,对应的P如下,设S为原始字符串:

通过观察上面的表,大家可以发现P[i]-1就是实际回文字串的长度。如果知道P,遍历一次就知道最长的回文子串。可以该如何计算P呢?这是这个算法最核心的部分。
具体可见下面的博客: Manacher's ALGORITHM: O(n)时间求字符串的最长回文子串

function getP5(str){  
    var p = [],   
        mx = 0,  
        id = 0;  
    var i;  
    // 将字符串转化为奇数长度获取到新的字符串  
    var newStr = '#';  
    var len = str.length;  
    for(i = 0;i<len;i++){  
        newStr += str[i]+'#';  
    }  
    var newLen = newStr.length;  
    for(i = 0;i<newLen;i++){  
        p[i] = 0;  
    }  
    // 时间复杂度为O(n),空间复杂度为O(1)获取到所有的子回文的长度值组成的数组  
    for (i = 0;i < newLen; i++) {  
        // mx表示当前边界值最大的回文子串的边界值  
        p[i] = mx > i ? Math.min(p[2*id-i], mx-i) : 1;  
        // 超出其半径的位置再做额外判断  
        while ((newStr[i + p[i]] == newStr[i - p[i]]) && newStr[i + p[i]]){  
            p[i]++;  
        }  
        // 获取到边界最大的回文子串的中心位置以及边界值,以保证后续迭代可以做以上快捷处理  
        if (i + p[i] > mx) {  
            id = i;  
            mx = id + p[i];  
        }  
    }  
    return p;  
}  

通过代码实现后,发现上面的博客对于其中的原理的分析并不十分准确。最基本的原则就是2个:
1、如果一个字符串(统一转化为了奇数长度)是回文串,中间字符位置为m。如果其子串S[i,i+x]是回文串,那么其相对于中间字符的对称子串S[2m-i,2m-(i+x)]也一定是回文串;
2、回文串一定是基于中心字符串对称的。也就是:S[i] == S[2m-i]。
然后分析下核心代码:

p[i] = mx > i ? Math.min(p[2*id-i], mx-i) : 1;  
        // 超出其半径的位置再做额外判断  
        while ((newStr[i + p[i]] == newStr[i - p[i]]) && newStr[i + p[i]]){  
            p[i]++;  
        }  
        // 获取到边界最大的回文子串的中心位置以及边界值,以保证后续迭代可以做以上快捷处理  
        if (i + p[i] > mx) {  
            id = i;  
            mx = id + p[i];  
        }  

分析后得知,mx并不是像博文中说的表示最大回文串的边界值,mx表示当前边界值最大的回文子串的边界值。由于超出边界值的部分无从判断是否还能与原串组成回文串,所以要额外判断。

今天是除夕日,祝福大家2017年能够事事顺利,健健康康,我也会继续更新相关方面的技术文章,希望能够带给读者们更多的知识!

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

推荐阅读更多精彩内容

  • 计算机二级C语言上机题库(南开版) 1.m个人的成绩存放在score数组中,请编写函数fun,它的功能是:将低于平...
    MrSunbeam阅读 6,354评论 1 42
  • 上一篇KMP算法之后好几天都没有更新,今天介绍最长回文子串。 首先介绍一下什么叫回文串,就是正着读和倒着读的字符顺...
    zero_sr阅读 2,290评论 2 8
  • 最长回文子串——Manacher 算法 1. 问题定义 最长回文字符串问题:给定一个字符串,求它的最长回文子串长度...
    林大鹏阅读 2,766评论 0 6
  • 字符串最长回文子串 题目描述: 给定一个字符串,求它的最长回文子串的长度。 分析和解法: 最容易想到的办法是枚举所...
    MinoyJet阅读 647评论 0 2
  • “颜值”这个词不知道从什么时候出现在生活中,然后像癌细胞一样蔓延在生活中的角角落落,好像干什么都是要看颜值。轻年一...
    雪岭阅读 1,257评论 1 3