最长回文字串

Posted zhibei

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了最长回文字串相关的知识,希望对你有一定的参考价值。

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

示例 1:

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

示例 2:

输入: "cbbd"
输出: "bb"

简单易于理解的解法
 1 class Solution(object):
 2     def longestPalindrome(self, s):
 3         """
 4         :type s: str
 5         :rtype: str
 6         """
 7         n = len(s)
 8         maxL, maxR, max = 0, 0, 0
 9         for i in range(n):
10             # 长度为偶数的回文字符串
11             start = i
12             end = i + 1
13             while start >= 0 and end < n:
14                 if s[start] == s[end]:
15                     if end - start + 1 > max:
16                         max = end - start + 1
17                         maxL = start
18                         maxR = end
19                     start -= 1
20                     end += 1
21                 else:
22                     break
23 
24             # 长度为奇数的回文子串
25             start = i - 1
26             end = i + 1
27             while start >= 0 and end < n:
28                 if s[start] == s[end]:
29                     if end - start + 1 > max:
30                         max = end - start + 1
31                         maxL = start
32                         maxR = end
33                     start -= 1
34                     end += 1
35                 else:
36                     break
37         return s[maxL:maxR + 1]
38 
39 s=abcba
40 str=Solution.longestPalindrome(1,s)
41 print(str)

查看资料找到了manacher算法 

    这个算法是以每一个字符为中心, 向两边发散,同时,用一个数组p来记录以每一个字符为中心的回文串的一半的长度.

     先看一下该算法的核心.

     确定以第i个字符为中心的回文串长度,建立在p[1:i-1]这个数组基础上

     假设, mx为当前字符串中已经确定的回文串的最有端. po为以mx为右端的回文串的中心,

     这时候,我们需要确定 第i个字符的位置是 > ma 还是 i <= mx

     如果 i <= mx 就是当前要确定的字符包含在前面的某个回文串里面, 回文串是对称的, 所以找到i 相对于po的对称位置 j, 

     接下来, 我们要做的就是判断, ①p[j] >= mx - i  or ②p[j] < mx - i

      在第一种情况, 就是说以i为中心的回文串还有向 mx右边伸展的可能, 然而那一部分还是未知数, 就的一步一步的去求解, 第二种情况, 就是 p[j] <= mx - i, 根据回文串的堆成性质, 可以确定

      p[i] = p[j]

     当i > mx的时候也是需要我们去一步一步的求解

     我在实现的时候只是给出最长回文串的一个, 

 1 class Solution:
 2     def longestPalindrome(self, s):
 3         """
 4         :type s: str
 5         :rtype: str
 6         """
 7         s1 = # + #.join(s) + #
 8         length = len(s1)
 9         p = [0] * length
10         i = 0
11         id, mx = 0, 0
12         while i < length:
13             if mx > i:
14                 p[i] = min(mx - i, p[2 * id - i])
15             else:
16                 p[i] = 1
17             while (i - p[i] >= 0) and (i + p[i] < length) and (s1[i - p[i]] == s1[i + p[i]]):
18                 p[i] += 1
19             if (i - 1 + p[i]) > mx:
20                 mx = i - 1 + p[i]
21                 id = i
22             i += 1
23         max_length = max(p)
24         index_id = p.index(max_length)
25         s = s1[index_id + 1 - max_length:index_id + max_length]
26         s2 = ‘‘
27         for i in s.split(#):
28             s2 += i
29         return s2
30 s=aa
31 str=Solution.longestPalindrome(1,s)
32 print(str)

第二种方法还不太明白,为什么要加‘#’号



以上是关于最长回文字串的主要内容,如果未能解决你的问题,请参考以下文章

最长对称子串 (最长回文字串) pta 7-12

最长回文字串

最长回文字串

java 最长回文字串

求最长回文字串

每日一题-Day12-最长回文字串