力扣HOT100之多维动态规划:5. 最长回文子串
这道题属于动态规划板块,但是我感觉这道题用动态规划相当晦涩难懂,看了一圈题解发现还是中心扩散法最好用,感觉这道题没有必要为了动态规划而动态规划,中心扩散法我感觉华南溜达虎的思路讲得很不错,非常通俗易懂,代码风格也很好,可以去看看他的视频。这道题就不用动态规划来做了,以后笔试面试遇到这道题就直接用中心扩散法了。
所谓中心扩散法,就是我们遍历字符串的每一个字符,并以该字符为中心,利用左右指针向两边扩散,如果左右指针指向的字符相等,则说明当前依然维护着回文子串的性质,我们可以进一步将左右指针向两边移动,当左右指针出现越界或者左右指针指向的字符不相等时退出循环。由于题目要求的是最长的回文子串,而不仅仅是最长的回文子串的长度,因此我们在循环中需要比较最长回文子串的长度与当前回文子串的长度,如果当前回文子串的长度更长,则更新最长回文子串的长度,并记下此时回文子串的起始下标。
我们注意到回文子串的长度可能是奇数,也可能是偶数,因此我们需要分类讨论:
1.回文子串长度为奇数的情况
例如字符串"abcbd"
,当我们从字符'c'
开始向两边扩散时,起始位置i = 2
,left = i, right = i;
,很显然,经过循环后,我们找到的最长回文子串为bcb
,但是对于字符串abba
,当我们从第二个'b'
开始从两边扩散时,如果还是按照left = i, right = i;
的初始化方式,一定找不到abba
这个最长回文子串,这就说明我们在选择每一个扩散中心时,当left
指针和right
指针指向同一个字符作为初始化操作时,只能找到长度为奇数的最长回文字串,我们在统计完奇数情况后,还应当及时统计偶数情况,以免遗漏情况。
2.回文子串长度为偶数的情况
例如字符串abccba
,当我们遍历到第一个a
时,应当让left
指向第一个a
,right
指向第一个b
,然后向两边扩散,很显然,在这种情况下得不到回文子串,遍历下一个字符。当遍历到第一个b
时,让left
指向第一个b
,right
指向第一个c
,很显然,在这种情况下得不到回文子串,遍历下一个字符。当遍历到第一个c
时,让left
指向第一个c
,right
指向第二个c
,在这种情况下,我们就得到了最长回文子串abccba
,长度为偶数。
在理解了奇数和偶数情况下各自的初始化方式后,我们就很容易写出代码了,我们通过一个for
循环遍历字符串s
所有的字符,分别讨论对应的奇数和偶数回文子串的情况,不断维护最长回文子串的长度及对应的起始下标,在for
循环结束后,直接调用s.substr()
获取最长回文子串并返回即可。
class Solution {
public:string longestPalindrome(string s) {int result_len = 0;int result_start = 0;for(int i = 0; i < s.size(); i++){//长度为奇数的回文子串int left = i, right = i;while(left >= 0 && right < s.size() && s[left] == s[right]){if(right - left + 1 > result_len){ //遇到更长的回文子串result_len = right - left + 1; //更新长度result_start = left; //记录起点}left--;right++;}//长度为偶数的回文子串left = i, right = i + 1;while(left >= 0 && right < s.size() && s[left] == s[right]){if(right - left + 1 > result_len){ //遇到更长的回文子串result_len = right - left + 1; //更新长度result_start = left; //记录起点}left--;right++;}}string result = s.substr(result_start, result_len);return result;}
};