给定一个字符串 s,找到 s 中最长的回文子串。你可以假设 s 的最大长度为 1000。

示例 1:

  1. 输入: "babad"
  2. 输出: "bab"
  3. 注意: "aba" 也是一个有效答案。

示例 2:

  1. 输入: "cbbd"
  2. 输出: "bb"

Given a string s, find the longest palindromic substring in s. You may assume that the maximum length of s is 1000.

Example 1:

  1. Input: "babad"
  2. Output: "bab"
  3. Note: "aba" is also a valid answer.

Example 2:

  1. Input: "cbbd"
  2. Output: "bb"

这道题让我们求最长回文子串,首先说下什么是回文串,就是正读反读都一样的字符串,比如 "bob", "level", "noon" 等等。那么最长回文子串就是在一个字符串中的那个最长的回文子串。LeetCode 中关于回文串的题共有五道,除了这道,其他的四道为 Palindrome NumberValidate PalindromePalindrome PartitioningPalindrome Partitioning II,我们知道传统的验证回文串的方法就是两个两个的对称验证是否相等,那么对于找回文字串的问题,就要以每一个字符为中心,像两边扩散来寻找回文串,这个算法的时间复杂度是 O(n*n),可以通过 OJ,就是要注意奇偶情况,由于回文串的长度可奇可偶,比如 "bob" 是奇数形式的回文,"noon" 就是偶数形式的回文,两种形式的回文都要搜索,对于奇数形式的,我们就从遍历到的位置为中心,向两边进行扩散,对于偶数情况,我们就把当前位置和下一个位置当作偶数行回文的最中间两个字符,然后向两边进行搜索,参见代码如下:

解法一:

  1. class Solution {
  2. public:
  3. string longestPalindrome(string s) {
  4. if (s.size() < 2) return s;
  5. int n = s.size(), maxLen = 0, start = 0;
  6. for (int i = 0; i < n - 1; ++i) {
  7. searchPalindrome(s, i, i, start, maxLen);
  8. searchPalindrome(s, i, i + 1, start, maxLen);
  9. }
  10. return s.substr(start, maxLen);
  11. }
  12. void searchPalindrome(string s, int left, int right, int& start, int& maxLen) {
  13. while (left >= 0 && right < s.size() && s[left] == s[right]) {
  14. --left; ++right;
  15. }
  16. if (maxLen < right - left - 1) {
  17. start = left + 1;
  18. maxLen = right - left - 1;
  19. }
  20. }
  21. };

我们也可以不使用子函数,直接在一个函数中搞定,我们还是要定义两个变量 start 和 maxLen,分别表示最长回文子串的起点跟长度,在遍历s中的字符的时候,我们首先判断剩余的字符数是否小于等于 maxLen 的一半,是的话表明就算从当前到末尾到子串是半个回文串,那么整个回文串长度最多也就是 maxLen,既然 maxLen 无法再变长了,计算这些就没有意义,直接在当前位置 break 掉就行了。否则就要继续判断,我们用两个变量left和right分别指向当前位置,然后我们先要做的是向右遍历跳过重复项,这个操作很必要,比如对于 noon,i在第一个o的位置,如果我们以o为最中心往两边扩散,是无法得到长度为4的回文串的,只有先跳过重复,此时left指向第一个o,right指向第二个o,然后再向两边扩散。而对于 bob,i在第一个o的位置时,无法向右跳过重复,此时 left 和 right 同时指向o,再向两边扩散也是正确的,所以可以同时处理奇数和偶数的回文串,之后的操作就是更新 maxLen 和 start 了,跟上面的操作一样,参见代码如下:

解法二:

  1. class Solution {
  2. public:
  3. string longestPalindrome(string s) {
  4. if (s.size() < 2) return s;
  5. int n = s.size(), maxLen = 0, start = 0;
  6. for (int i = 0; i < n;) {
  7. if (n - i <= maxLen / 2) break;
  8. int left = i, right = i;
  9. while (right < n - 1 && s[right + 1] == s[right]) ++right;
  10. i = right + 1;
  11. while (right < n - 1 && left > 0 && s[right + 1] == s[left - 1]) {
  12. ++right; --left;
  13. }
  14. if (maxLen < right - left + 1) {
  15. maxLen = right - left + 1;
  16. start = left;
  17. }
  18. }
  19. return s.substr(start, maxLen);
  20. }
  21. };

此题还可以用动态规划 Dynamic Programming 来解,根 Palindrome Partitioning II 的解法很类似,我们维护一个二维数组 dp,其中 dp[i][j] 表示字符串区间 [i, j] 是否为回文串,当 i = j 时,只有一个字符,肯定是回文串,如果 i = j + 1,说明是相邻字符,此时需要判断 s[i] 是否等于 s[j],如果i和j不相邻,即 i - j >= 2 时,除了判断 s[i] 和 s[j] 相等之外,dp[i + 1][j - 1] 若为真,就是回文串,通过以上分析,可以写出递推式如下:

dp[i, j] = 1                                               if i == j

= s[i] == s[j]                                if j = i + 1

= s[i] == s[j] && dp[i + 1][j - 1]    if j > i + 1

这里有个有趣的现象就是如果我把下面的代码中的二维数组由 int 改为 vector<vector<int>> 后,就会超时,这说明 int 型的二维数组访问执行速度完爆 std 的 vector 啊,所以以后尽可能的还是用最原始的数据类型吧。

解法三:

  1. class Solution {
  2. public:
  3. string longestPalindrome(string s) {
  4. if (s.empty()) return "";
  5. int dp[s.size()][s.size()] = {0}, left = 0, right = 0, len = 0;
  6. for (int i = 0; i < s.size(); ++i) {
  7. dp[i][i] = 1;
  8. for (int j = 0; j < i; ++j) {
  9. dp[j][i] = (s[i] == s[j] && (i - j < 2 || dp[j + 1][i - 1]));
  10. if (dp[j][i] && len < i - j + 1) {
  11. len = i - j + 1;
  12. left = j;
  13. right = i;
  14. }
  15. }
  16. }
  17. return s.substr(left, right - left + 1);
  18. }
  19. };

最后要来的就是大名鼎鼎的马拉车算法 Manacher's Algorithm,这个算法的神奇之处在于将时间复杂度提升到了 O(n) 这种逆天的地步,而算法本身也设计的很巧妙,很值得我们掌握,参见我另一篇专门介绍马拉车算法的博客 Manacher's Algorithm 马拉车算法,代码实现如下:

解法四:

  1. class Solution {
  2. public:
  3. string longestPalindrome(string s) {
  4. string t ="$#";
  5. for (int i = 0; i < s.size(); ++i) {
  6. t += s[i];
  7. t += '#';
  8. }
  9. int p[t.size()] = {0}, id = 0, mx = 0, resId = 0, resMx = 0;
  10. for (int i = 1; i < t.size(); ++i) {
  11. p[i] = mx > i ? min(p[2 * id - i], mx - i) : 1;
  12. while (t[i + p[i]] == t[i - p[i]]) ++p[i];
  13. if (mx < i + p[i]) {
  14. mx = i + p[i];
  15. id = i;
  16. }
  17. if (resMx < p[i]) {
  18. resMx = p[i];
  19. resId = i;
  20. }
  21. }
  22. return s.substr((resId - resMx) / 2, resMx - 1);
  23. }
  24. };

最长回文子串 C++实现 java实现 leetcode系列(五)的更多相关文章

  1. [LeetCode]最长回文子串 java

    题目: 给定一个字符串 s,找到 s 中最长的回文子串.你可以假设 s 的最大长度为1000. 示例 1: 输入: "babad" 输出: "bab" 注意: ...

  2. Java实现 LeetCode 5 最长回文子串

    5. 最长回文子串 给定一个字符串 s,找到 s 中最长的回文子串.你可以假设 s 的最大长度为 1000. 示例 1: 输入: "babad" 输出: "bab&quo ...

  3. 最长回文子串(Longest Palindromic Substring)

    这算是一道经典的题目了,最长回文子串问题是在一个字符串中求得满足回文子串条件的最长的那一个.常见的解题方法有三种: (1)暴力枚举法,以每个元素为中心同时向左和向右出发,复杂度O(n^2): (2)动 ...

  4. lintcode :Longest Palindromic Substring 最长回文子串

    题目 最长回文子串 给出一个字符串(假设长度最长为1000),求出它的最长回文子串,你可以假定只有一个满足条件的最长回文串. 样例 给出字符串 "abcdzdcab",它的最长回文 ...

  5. leetcode 5 :Longest Palindromic Substring 找出最长回文子串

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

  6. 最长回文子串(Longest Palindromic Substring)-DP问题

    问题描述: 给定一个字符串S,找出它的最大的回文子串,你可以假设字符串的最大长度是1000,而且存在唯一的最长回文子串 . 思路分析: 动态规划的思路:dp[i][j] 表示的是 从i 到 j 的字串 ...

  7. leetcode刷题五<最长回文子串>

    下面是题目的描述 给定一个字符串 s,找到 s 中最长的回文子串.你可以假设 s 的最大长度为 . 示例 : 输入: "babad" 输出: "bab" 注意: ...

  8. LeetCode(5):最长回文子串

    Medium! 题目描述: 给定一个字符串 s,找到 s 中最长的回文子串.你可以假设 s 长度最长为1000. 示例: 输入: "babad" 输出: "bab&quo ...

  9. HDU 4745 Two Rabbits (2013杭州网络赛1008,最长回文子串)

    Two Rabbits Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others)Tota ...

随机推荐

  1. ES内存持续上升问题定位

      https://discuss.elastic.co/t/memory-usage-of-the-machine-with-es-is-continuously-increasing/23537/ ...

  2. oracle故障处理之删除大表空间hang住

    背景 数据库分区表数据越来越大,需要对过期话的数据进行迁移,以及大的分区表需要进行数据的清理和删除,达到释放磁盘空间的目的. 问题说明 环境:linux 6.X 数据库:oracle 11.2.0.4 ...

  3. 对ajax和jsp渲染的一些感想

    一个多月来,也算是做了两个javaweb的项目,但是对于用哪种方式渲染页面一直有些抉择不定,第一个项目用的ajax,发现各种对htmlDom节点的操作十分的繁琐,需要找到数据填充的节点并添加id值,引 ...

  4. C# Distanct List集合

    简单一维集合的使用 List<int> ages = new List<int> { 21, 46, 46, 55, 17, 21, 55, 55 }; List<str ...

  5. struts2的DTD配置文件

    新手可以看看,高手可以跳过…… 最近在学习struts2这个框架,自己也动手写过一些DTD文件,所以很好struts2这个DTD文件是怎么写的,接下来就一个一个的分析 根元素是struts,然后又4个 ...

  6. MySQL构造测试数据

    构造测试数据(笛卡尔积,6 次100 万) create table t1(id int, val varchar(80)); set @i := 0;create table tmp as sele ...

  7. 路飞学城Python-Day42

    前端                                                                                                  ...

  8. 路飞学城Python-Day37

    36-多表查询练习 37-权限管理 1.创建账号 本地账号 create user 'panda' @ 'loacalhost' inentified by'123' 远程账号 create user ...

  9. Docker-镜像的操作命令

    2.镜像在Ubuntu中的一些命令 (1)docker image ls 列出镜像 能够罗列出docker中所以的镜像所在的仓库.镜像标签.镜像ID.镜像的创建日期.镜像的大小等等信息. (2)doc ...

  10. CSS Grid(CSS网格)

    Grid被设计来做一些Flexbox不能做的事情,所以不是被设计来取代Flexbox的. flexbox 一维的 Grid 二维的 总结:  Grid Items作用在Grid Container的直 ...