一天一道LeetCode系列

(一)题目

Given a string S, find the longest palindromic substring in S. You may

assume that the maximum length of S is 1000, and there exists one

unique longest palindromic substring.

题意:求一个字符串的最长回文字串。

(二)解题

1.中心扩展法

看到这个题首先想到的就是中心扩展法,遍历每一个字符,然后以该字符为中心向四周扩展,这种方法的时间复杂度为O(N^2),但是需要注意的是,奇数和偶数回文字符串需要区别对待,如aba和abba都是回文字符串。

class Solution {
public:
    string longestPalindrome(string s) {
        int max1=0;//奇数最长子串
        int max2=0;//偶数最长子串
        int idx1=0;//奇数最长子串的中心字符
        int idx2=0;//偶数最长子串的中心字符
        string result;
        for(int i = 0 ; i < s.length() ; ++i)
        {
            int j = i;
            int z = i;
            int count1=0;
            int count2=0;
            //计算奇数最长回文字符串
            while((++z<s.length()) && (--j>=0) && (s[z] == s[j]))
            {
                count1+=2;
                if(count1 > max1)
                {
                    max1=count1;
                    idx1 = i;
                }
            }
            //计算偶数最长回文字符串
            j = i;
            z = i+1;
            while((z<s.length()) && (j>=0) &&(s[z] == s[j]))
            {
                count2+=2;
                if(count2 > max2)
                {
                    max2=count2;
                    idx2 = i;
                }
                z++;
                j--;
            }
        }
        if(max1+1>max2) result = s.substr(idx1-max1/2,max1+1);
        else result = s.substr(idx2-max2/2+1,max2);
        return result;
    }
};

2.中心扩展法的优化

区分奇数和偶数显得程序比较臃肿,我们可以利用“改造“字符串来避免这种情况。如aba改成#a#b#a#,abba改成#a#b#b#a,这样就不用区分奇偶了。

class Solution {
public:
    string longestPalindrome(string s) {
        int max=0;
        int idx=0;
        string temp[2005];
        //改造字符串
        int j = 0;
        for(int i = 0; i < s.length() ; ++i)
        {
            temp[j++] = '#';
            temp[j++] = s[i];
        }
        temp[j++] = '#';
        temp[j] = '\0';
        for(int i = 0 ; i <2*s.length()+1 ; ++i)
        {
            int j = i;
            int z = i;
            int count=0;
            //计算奇数最长回文字符串
            while((++z<(2*s.length()+1)) && (--j>=0) && (temp[z] == temp[j]))
            {
                count++;
                if(count > max)
                {
                    max=count;
                    idx = i;
                }
            } 

        }
        return s.substr((idx-max)/2,max);
    }
};

3.动态规划法

DP算法的思想就是记录每一个回文子串的位置,在每一次判断是否为回文子串的时候先判断它的子串是不是回文,例如,用map[i][j]记录i到j为回文数组,如果这个时候s[i-1]==s[j+1],那么就能在O(1)时间内判断[i-1,j+1]是否为回文了。

动态规划法的时间复杂度为O(n^2).

class Solution {
public:
    string longestPalindrome(string s) {
        int len = s.length();
        int idx = 0;//记录最长回文字符的开始处
        int max = 1;//记录最长回文字符的长度
        int map[1000][1000] = {0};//记录i到j是否为回文子串
        for(int i = 0 ; i < len ; ++i)
        {
            map[i][i] = 1;//初始化长度为1的回文子串
        }
        for(int i = 0 ; i < len ; ++i)
        {
            if(s[i] == s[i+1])//初始化长度为2的子串
            {
                map[i][i+1] = 1;
                idx = i;
                max = 2;
            }
        }

        for(int plen = 3 ; plen <= len ; plen++)//从长度为3开始算起
        {//plen代表当前判断的回文子串的长度
            for(int j = 0 ; j < len - plen +1 ; j++)
            {
                int z = plen+j-1;//z为回文子串的尾序号
                if (s[j] == s[z] && map[j+1][z-1]) {
                //O(1)时间内判断j到z是否回文
                    map[j][z] = 1;
                    idx = j;
                    max = plen;
                }
            }
        }
        return s.substr(idx,max);//返回子串
    }
};

4.经典的Manacher算法,O(n)复杂度

算法步骤:

step1:跟解法2一样,改造字符串:

abba –> $#a#b#b#a#

注:加’$’是为了避免处理越界问题

step2:用p[i]记录以i为中心点的回文字符串长度

改造后的字符串:$#a#b#b#a#

p[]数组的值:121242121

注:p[i]-1 = 源字符串中回文子串长度

step3:利用DP的思想来求解p[]

利用中心扩展法求以i为中心的最长回文串

while(i+p[i] < temps.length() && temps[i-p[i]] == temps[i+p[i]]) p[i]++;

利用p[],mx,id记录的已有回文字符串的长度来避免大量重复的匹配

if(mx > i) p[i] = p[2*id-i] < (mx-i) ? p[2*id-i]:(mx-i);

注:p[2*id-i]为i关于j对称的点的回文串长度

mx为i之前的回文串延伸到右边的最长位置,id为该回文串的中间值

class Solution {
public:
    string longestPalindrome(string s) {
        string temps;
        temps+="$#";//加$是为了避免处理越界情况,减小时间复杂度
        for(int i = 0 ; i < s.length() ; ++i)//
        {
            temps+=s[i];
            temps+="#";
        }
        int *p = new int[temps.length()];//p[i]记录以i为中心的回文串长度
        memset(p, 0, sizeof(p));

        int max = 0,idx = 0;//max记录最长回文串的长度,idx记录最长回文串的中心位置
        int mx = 0,id = 0;//mx记录i之前的最长回文串延伸到最右边的位置,id记录该字符串的中心位置
        for(int i = 1; i < temps.length() ; ++i)
        {
            if(mx > i)
            {
                p[i] = p[2*id-i] < (mx-i) ? p[2*id-i]:(mx-i);
            }
            else
                p[i] = 1;

            while(i+p[i] < temps.length() && temps[i-p[i]] == temps[i+p[i]])
            {
                p[i]++;
            }
            if(i+p[i]>mx)
            {
                id = i;
                mx = i+p[i];
                if(p[i]>max)
                {
                    max = p[i];
                    idx = id;
                }
            }
        }
        max--;
        return s.substr((idx-max)/2,max);
    }
};

该算法的时间复杂度为O(n)。

以上代码在LeetCode中均Accepted。

【一天一道LeetCode】#5 Longest Palindromic Substring的更多相关文章

  1. LeetCode(4) || Longest Palindromic Substring 与 Manacher 线性算法

    LeetCode(4) || Longest Palindromic Substring 与 Manacher 线性算法 题记 本文是LeetCode题库的第五题,没想到做这些题的速度会这么慢,工作之 ...

  2. Leetcode 5. Longest Palindromic Substring(最长回文子串, Manacher算法)

    Leetcode 5. Longest Palindromic Substring(最长回文子串, Manacher算法) Given a string s, find the longest pal ...

  3. 求最长回文子串 - leetcode 5. Longest Palindromic Substring

    写在前面:忍不住吐槽几句今天上海的天气,次奥,鞋子里都能养鱼了...裤子也全湿了,衣服也全湿了,关键是这天气还打空调,只能瑟瑟发抖祈祷不要感冒了.... 前后切了一百零几道leetcode的题(sol ...

  4. LeetCode 5 Longest Palindromic Substring(最长子序列)

    题目来源:https://leetcode.com/problems/longest-palindromic-substring/ Given a string S, find the longest ...

  5. 【JAVA、C++】LeetCode 005 Longest Palindromic Substring

    Given a string S, find the longest palindromic substring in S. You may assume that the maximum lengt ...

  6. leetcode:Longest Palindromic Substring(求最大的回文字符串)

    Question:Given a string S, find the longest palindromic substring in S. You may assume that the maxi ...

  7. [LeetCode][Python]Longest Palindromic Substring

    # -*- coding: utf8 -*-'''__author__ = 'dabay.wang@gmail.com'https://oj.leetcode.com/problems/longest ...

  8. 【LeetCode】Longest Palindromic Substring 解题报告

    DP.KMP什么的都太高大上了.自己想了个朴素的遍历方法. [题目] Given a string S, find the longest palindromic substring in S. Yo ...

  9. [LeetCode] 5. Longest Palindromic Substring 最长回文子串

    Given a string s, find the longest palindromic substring in s. You may assume that the maximum lengt ...

  10. 最长回文子串-LeetCode 5 Longest Palindromic Substring

    题目描述 Given a string S, find the longest palindromic substring in S. You may assume that the maximum ...

随机推荐

  1. Android图表库MPAndroidChart(七)—饼状图可以再简单一点

    Android图表库MPAndroidChart(七)-饼状图可以再简单一点 接上文,今天实现的是用的很多的,作用在统计上的饼状图,我们看下今天的效果 这个效果,我们实现,和之前一样的套路,我先来说下 ...

  2. VS2010每次调试都出现“此项目已经过期”提示

    问题描述   最近因为项目需要,开发平台从VS2005切换成了VS2010,把一些老项目也转换到VS2010平台,因为是从低到高升级,微软还是做了很多兼容,基本上可以无缝切换,编译调试也基本正常,但是 ...

  3. Android N(7.0) 被美翻的新特性!

    Tamic 专注移动开发!更多文章请关注 Csdn: http://blog.csdn.net/sk719887916/article/details/52612444 $ http://www.ji ...

  4. Eric5 for Python 3.3.3安装指南

    一言蔽之,搭配是关键.以32位Window为例,先后安装: 1.PyQt PyQt4-4.10.3-gpl-Py3.3-Qt4.8.5-x32.exe http://www.riverbankcomp ...

  5. 关于JQuery中的ajax请求或者post请求的回调方法中的操作执行或者变量修改没反映的问题

    前段时间做一个项目,而项目中所有的请求都要用jquery 中的ajax请求或者post请求,但是开始处理一些简单操作还好,但是自己写了一些验证就出现问题了,比如表单提交的时候,要验证帐号的唯一性,所以 ...

  6. SceneKit:简单的3D游戏场景搭建

    SceneKit是Apple用来开发休闲3D游戏的框架,不同于底层的OpenGL库,你仅仅需要很少的代码就可以快速看到实际的3D场景效果.下面简单的聊聊搭建一个3D游戏场景需要做的事情. 首先你必须用 ...

  7. lucene查询索引库、分页、过滤、排序、高亮

    2.查询索引库 插入测试数据 xx.xx. index. ArticleIndex @Test public void testCreateIndexBatch() throws Exception{ ...

  8. frameset 与frame 设置的技巧

    今天来写点不一样的.如下图: 实现的效果就是原生的类似于导航形式的frameset. frameset 注意: 包含frameset的网页应该只是作为框架而存在,所以不能有body标签. 这个标签可以 ...

  9. Android简易实战教程--第二十五话《网络图片查看器》

    访问网络已经有了很成熟的框架.这一篇只是介绍一下HttpURLConnection的简单用法,以及里面的"注意点".这一篇可以复习或者学习HttpURLConnection.han ...

  10. 第一行代码阅读笔记----显示隐式Intent的基本用法

    1.显示Intent意图明显,通过Intent启动另外一个活动,这是安卓中各组件进行交互的一种重要方式.一般用于启动活动,启动服务,发送广播等场景. 实现方法,这里我只说思路,实践还是要自己实操才能明 ...