一、题目:最长回文子串
二、题目解析:
题目解析:
- 示例1和2表明最长子串不能重复,示例3表明必须要连续,可以利用滑动窗口的思想,定义两个指针i,j和一个set,一个指针i不断往前遍历,判断set中是否有当前值,有移动j,并移除set中的元素,直到集合中不包含当前元素,添加到set中,直到循环遍历结束。
- 回文子串就是说一个字符串从前往后读和从后往前读都是一样的,主要是利用中心扩散的思想,把每一个字符当做中心,检查它的左边和右边的字符是否是相等的,相等则说明是回文子串,但要注意特殊情况,不是所有的回文子字符串都有一个中心点 ,比如cabbad 假设以b为中心a并不等于b,所以这种情况最大回文是b本身,实际上它的中心应该是在b和b之间 (abba),所以这里分两种情况,一种是中间有一个中心点朝两边扩散,另一种情况是中间并没有中心点朝两边扩散。
- 解题步骤:
3.1 如果字符串长度小于2,直接返回原字符串
3.2 定义两个变量,一个start存储当前查找到的最大回文串的起始位置,另一个maxLength记录字符串的的长度(终止位置就是start+maxLength)
3.3 创建一个helper function,判断左边和右边是否越界,同时最左边的字符是否等于最右边的字符。
如果以上2个条件都满足,则判断是否需要更新回文字符串最大长度及最大字符串的起始位置。
然后将left–,right++,继续判断,直到不满足两个个条件之一。
3.4 遍历字符串,每个位置调用helper function两遍,第一遍检查i-1,i+1,第二遍检查i,i+1(为甚么要检查2遍?) 两种情况啊 babad,cabbad
图示加深理解:
三、代码如下:
public class Solution {
private int start = 0, maxLength = 1; //maxLength = 1的原因是特殊情况,ab 最大回文子串长度不是0,应该是1,所以maxLength要初始化为1
public String longestPalindrome(String s) {
if (s.length() < 2) { //处理边界问题
return s;
}
for (int i = 0; i < s.length(); i++) {
expandAroundCenter(s, i - 1, i + 1);
expandAroundCenter(s, i, i + 1);
}
return s.substring(start, start + maxLength);
}
//从中心向两边扩展
private void expandAroundCenter(String s, int left, int right) {
//判断保证左边和右边不能越界,并且两边字符都相等
while (left >= 0 && right < s.length() && s.charAt(left) == s.charAt(right)) {
if (right - left + 1 > maxLength) { //判断是否需要更新最大长度 为啥right-left+1? [a,b,a] 2-0+1 ( right-left+1 )
//进到这里说明找到了一个更大的回文字符串
maxLength = right - left + 1; //更新maxLength
start = left; //更新start
}
left--;
right++;
}
}
}
四、测试
五、结束