1 Star 0 Fork 0

iint/notes

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
该仓库未声明开源许可证文件(LICENSE),使用请关注具体项目描述及其代码上游依赖。
克隆/下载
leetcode5.LongestPalindromicSubstring.md 6.73 KB
一键复制 编辑 原始数据 按行查看 历史
nuaazs 提交于 4年前 . first

## 5. Longest Palindromic Substring

Given a string s, return the longest palindromic substring in s.

Example 1:

Input: s = "babad"
Output: "bab"
Note: "aba" is also a valid answer.

Example 2:

Input: s = "cbbd"
Output: "bb"

Example 3:

Input: s = "a"
Output: "a"

Example 4:

Input: s = "ac"
Output: "a"

Constraints:

  • 1 <= s.length <= 1000
  • s consist of only digits and English letters (lower-case and/or upper-case),

Solution

Approach 1: Longest Common Substring

Common mistake

Some people will be tempted to come up with a quick solution, which is unfortunately flawed (however can be corrected easily):

Reverse SS and become S'S′. Find the longest common substring between SS and S'S′, which must also be the longest palindromic substring.

This seemed to work, let’s see some examples below.

For example, SS = "caba", S'S′ = "abac".

The longest common substring between SS and S'S′ is "aba", which is the answer.

Let’s try another example: SS = "abacdfgdcaba", S'S′ = "abacdgfdcaba".

The longest common substring between SS and S'S′ is "abacd". Clearly, this is not a valid palindrome.

Algorithm

We could see that the longest common substring method fails when there exists a reversed copy of a non-palindromic substring in some other part of SS. To rectify this, each time we find a longest common substring candidate, we check if the substring’s indices are the same as the reversed substring’s original indices. If it is, then we attempt to update the longest palindrome found so far; if not, we skip this and find the next candidate.

This gives us an O(n^2) Dynamic Programming solution which uses O(n^2)space (could be improved to use O(n) space). Please read more about Longest Common Substring here.


Approach 2: Brute Force

The obvious brute force solution is to pick all possible starting and ending positions for a substring, and verify if it is a palindrome.

Complexity Analysis

  • Time complexity : O(n^3). Assume that n is the length of the input string, there are a total of<img alt="image-20210306113932464">such substrings (excluding the trivial solution where a character itself is a palindrome). Since verifying each substring takes O(n)time, the run time complexity is O(n^3).

  • Space complexity : O(1).


Approach 3: Dynamic Programming

To improve over the brute force solution, we first observe how we can avoid unnecessary re-computation while validating palindromes. Consider the case "ababa". If we already knew that "bab" is a palindrome, it is obvious that "ababa" must be a palindrome since the two left and right end letters are the same.

We define P(i,j)as following:

<img alt="image-20210306114042236">

Therefore,

<img alt="image-20210306114115341">

The base cases are:

<img alt="image-20210306114125042">

This yields a straight forward DP solution, which we first initialize the one and two letters palindromes, and work our way up finding all three letters palindromes, and so on...

Complexity Analysis

  • Time complexity : O(n^2). This gives us a runtime complexity of O(n^2).
  • Space complexity : O(n^2). It uses O(n^2)space to store the table.

Additional Exercise

Could you improve the above space complexity further and how?


Approach 4: Expand Around Center

In fact, we could solve it in O(n^2) time using only constant space.

We observe that a palindrome mirrors around its center. Therefore, a palindrome can be expanded from its center, and there are only 2n - 1such centers.

You might be asking why there are 2n - 1 but not n centers? The reason is the center of a palindrome can be in between two letters. Such palindromes have even number of letters (such as "abba") and its center are between the two 'b's.

public String longestPalindrome(String s){
    if (s == null || s.length() < 1) return "";
    int start = 0, end = 0;
    for (int i =0;i<s.length();i++){
        int len1 = expandAroundCenter(s,i,i);
        int len2 = expandAroundCenter(s,i,i+1);
        int len = Math.max(len1,len2);
        if(len>end-start){
            start = i - (len -1)/2;
            end = i + len/2;
        }
      }
    return s.substring(start, end+1);
}
private int expandAroundCenter(String s, int left, int right){
    int L = left,R = right;
    while(L>=0 && R<s.length() && s.charAt(L) == s.charAt(R)){
        L--;
        R++;
    }
    return R-L-1
}
class Solution{
    public:
    string longestPalindrome(string s){
        int start=0; int end=0;
        if(s.size()==0)
            return "";
        else{
            for (int center=0;center<s.size();center++){
                int len1=PaliLength(center,center,s);
                int len2=PaliLength(center,center+1,s);
                if(len1>end-start){
                    start = center-len1/2;
                    end=center+len1/2;
                }
                if(len2>end-start){
                    start=center+1-len2/2;
                    end=center+len2/2;
                }
                
            }
        }
        return s.substr(start,end-start+1);
    }
    
    int PaliLength(int L, int R, string s){
        int len=0;
        while(L>=0 && R<s.size()){
            if(s[L]==s[R]){
                len=R-L+1;
                L--;
                R++;
            }
            else
                break;
        }
        return len;
    }
};

Python的:不太好

class Solutioin:
    def longestPalindrome(self,s:str)->str:
        m = ''
        for i in range(len(s)):
            for j in range(len(s),i,-1):
                if len(m) >= j-i:
                    break
                elif s[i:j]==s[i:j][::-1]
                	m=s[i:j]
                    break
        return m

Complexity Analysis

  • Time complexity : O(n^2). Since expanding a palindrome around its center could take O(n) time, the overall complexity is O(n^2) .

  • Space complexity : O(1).


Approach 5: Manacher's Algorithm

There is even an O(n) algorithm called Manacher's algorithm, explained here in detail. However, it is a non-trivial algorithm, and no one expects you to come up with this algorithm in a 45 minutes coding session. But, please go ahead and understand it, I promise it will be a lot of fun.

Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/iint/notes.git
git@gitee.com:iint/notes.git
iint
notes
notes
master

搜索帮助