题目描述

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

示例 1:

  1. 输入: "babad"

  2. 输出: "bab"

  3. 注意: "aba" 也是一个有效答案。

示例 2:

  1. 输入: "cbbd"

  2. 输出: "bb"

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/longest-palindromic-substring
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

这是一道最长回文的题目,要我们求出给定字符串的最大回文子串。

5. 最长回文子串_回文子串

解决这类问题的核心思想就是两个字“延伸”,具体来说

  • 如果一个字符串是回文串,那么在它左右分别加上一个相同的字符,那么它一定还是一个回文串

  • 如果一个字符串不是回文串,或者在回文串左右分别加不同的字符,得到的一定不是回文串

事实上,上面的分析已经建立了大问题和小问题之间的关联, 基于此,我们可以建立动态规划模型。

我们可以用 dp[i][j] 表示 s 中从 i 到 j(包括 i 和 j)是否可以形成回文, 状态转移方程只是将上面的描述转化为代码即可:

  1. if (s[i] === s[j] && dp[i + 1][j - 1]) {

  2. dp[i][j] = true;

  3. }

 

5. 最长回文子串_回文串_02

base case就是一个字符(轴对称点是本身),或者两个字符(轴对称点是介于两者之间的虚拟点)。

5. 最长回文子串_回文串_03

 

关键点

  • ”延伸“(extend)

代码

  1. /*

  2. * @lc app=leetcode id=5 lang=javascript

  3. *

  4. * [5] Longest Palindromic Substring

  5. */

  6. /**

  7. * @param {string} s

  8. * @return {string}

  9. */

  10. var longestPalindrome = function(s) {

  11. // babad

  12. // tag : dp

  13. if (!s || s.length === 0) return "";

  14. let res = s[0];

  15.  

  16. const dp = [];

  17.  

  18. // 倒着遍历简化操作, 这么做的原因是dp[i][..]依赖于dp[i + 1][..]

  19. for (let i = s.length - 1; i >= 0; i--) {

  20. dp[i] = [];

  21. for (let j = i; j < s.length; j++) {

  22. if (j - i === 0) dp[i][j] = true;

  23. // specail case 1

  24. else if (j - i === 1 && s[i] === s[j]) dp[i][j] = true;

  25. // specail case 2

  26. else if (s[i] === s[j] && dp[i + 1][j - 1]) {

  27. // state transition

  28. dp[i][j] = true;

  29. }

  30.  

  31. if (dp[i][j] && j - i + 1 > res.length) {

  32. // update res

  33. res = s.slice(i, j + 1);

  34. }

  35. }

  36. }

  37.  

  38. return res;

  39. };

相关题目

  • [516.longest-palindromic-subsequence]