题目
定义
回文,就是一个字符串正着念和反着念都一样,比如“abcba”,比如“上海自来水来自还上”,比如“明天到操场….”。
对于这道算法题研究的英文字符串来说,有奇数回文和偶数回文两种情况。奇数回文是指回文字符串是奇数个,比如abcba;偶数回文是指回文字符串是偶数个,比如abba。在设计算法的时候应该把这两种情况都考虑在内。
Given a string s, find the longest palindromic substring in s. You may assume that the maximum length of s is 1000.
Example 1:
Input: “babad”
Output: “bab”
Note: “aba” is also a valid answer.
Example 2:
Input: “cbbd”
Output: “bb”
解法一 中心拓展
以某个元素为中心,分别计算偶数长度的回文最大长度和奇数长度的回文最大长度。时间复杂度O(n^2),空间O(1)。
class Solution {
public:
string longestPalindrome(string s) {
const int len = s.size();
if(len <= 1)return s;
int start, maxLen = 0;
for(int i = 1; i < len; i++)
{
//寻找以i-1,i为中点偶数长度的回文
int low = i-1, high = i;
while(low >= 0 && high < len && s[low] == s[high])
{
low--;
high++;
}
if(high - low - 1 > maxLen)
{
maxLen = high - low -1;
start = low + 1;
}
//寻找以i为中心的奇数长度的回文
low = i- 1; high = i + 1;
while(low >= 0 && high < len && s[low] == s[high])
{
low--;
high++;
}
if(high - low - 1 > maxLen)
{
maxLen = high - low -1;
start = low + 1;
}
}
return s.substr(start, maxLen);
}
};
解法二 动态规划
在动态规划中保存最长回文的长度及起点即可
class Solution {
public:
string longestPalindrome(string s) {
const int len = s.size();
if(len <= 1)return s;
bool dp[len][len];//dp[i][j]表示s[i..j]是否是回文
memset(dp, 0, sizeof(dp));
int resLeft = 0, resRight = 0;
dp[0][0] = true;
for(int i = 1; i < len; i++)
{
dp[i][i] = true;
dp[i][i-1] = true;//这个初始化容易忽略,当k=2时要用到
}
for(int k = 2; k <= len; k++)//枚举子串长度
for(int i = 0; i <= len-k; i++)//枚举子串起始位置
{
if(s[i] == s[i+k-1] && dp[i+1][i+k-2])
{
dp[i][i+k-1] = true;
if(resRight-resLeft+1 < k)
{
resLeft = i;
resRight = i+k-1;
}
}
}
return s.substr(resLeft, resRight-resLeft+1);
}
};