回文子串子序列
回文子串
回文字符串 是正着读和倒过来读一样的字符串。
子字符串 是字符串中的由连续字符组成的一个序列。
https://leetcode.cn/problems/palindromic-substrings/
如果是把字符串分割成子串,需要对这种分割方案进行逻辑判断,只能用回溯或背包
如果只是找到字符串的所有子串,可以使用循环暴力搜索
- 循环暴力搜索
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23public int countSubstrings(String s) {
int result = 0;
// 遍历不同起点与终点暴力搜索
for (int i=0; i<s.length(); i++) {
for (int j=i+1; j<=s.length(); j++) {
String str = s.substring(i, j);
if (this.isSame(str)) {
result++;
}
}
}
return result;
}
private boolean isSame(String s) {
int mid = (s.length() - 1) / 2;
for (int i=0; i<=mid; i++) {
if (s.charAt(i) != s.charAt(s.length()-1-i)) {
return false;
}
}
return true;
} - 动态规划
1
2
3
4
5
6
7
8
9
10
11
12
13
141. dp[i][j] [i, j]是否是回文子串
2. s[i] != s[j] dp[i][j] = false
s[i] == s[j]
if (i==j || j-i==1) dp[i][j] = true
else dp[i][j] = dp[i+1][j-1]
3. dp[i][j] = false
4. dp[i][j] 需要依赖dp[i+1][j-1]
i+1 在i的下方 j-1在j的左侧
所以遍历顺序是先下后上,先左后右
5. "abc"
j=0 j=1 j=2
i=0 true false false
i=1 flase true false
i=2 false false true
1 | public int countSubstrings(String s) { |
最长回文子序列
https://leetcode.cn/problems/longest-palindromic-subsequence/
子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。
- 动态规划
1
2
3
4
5
6
7
8
9
10
11
12
131. dp[i][j] [i, j]区间子序列的最大长度
2. s[i] == s[j] dp[i][j] = dp[i+1][j-1] + 2
s[i] != s[j] dp[i][j] = Math.max(dp[i+1][j], dp[i][j-1])
3. i==j 时,dp[i][j] 不能使用dp[i+1][j-1] + 2递推得到,必须单独初始化为1 其他初始化为0
4. dp[i][j] 依赖 dp[i+1][j-1] 左下方 dp[i+1][j]下方 dp[i][j-1]左侧
从下到上,从左到右的计算
5. s = "bbbab"
b b b a b
b 1 2 3 3 4
b 1 2 2 3
b 1 1 3
a 1 1
b 1
1 | public int longestPalindromeSubseq(String s) { |
- 回溯 超时
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33private int maxLength;
public int longestPalindromeSubseq(String s) {
this.backtrack(s, 0, "");
return this.maxLength;
}
// 子序列问题,与子集问题一致,组合问题的升级版
// 组合问题只收集叶子节点,子集问题收集所有节点
// 子集问题与子序列问题的回溯参数与返回值一致,添加path参数自动回溯
private void backtrack(String s, int startIndex, String path) {
// 保存结果
if (this.isSame(path)) {
this.maxLength = Math.max(this.maxLength, path.length());
}
// 利用for循环条件结束递归
for (int i=startIndex; i<s.length(); i++) {
// 从下一个字符开始遍历
backtrack(s, i+1, path+s.charAt(i));
}
}
private boolean isSame(String s) {
if (s.length() == 0) {
return false;
}
int mid = (s.length() - 1) / 2;
for (int i=0; i<=mid; i++) {
if (s.charAt(i) != s.charAt(s.length()-1-i)) {
return false;
}
}
return true;
}
最长回文子串
https://leetcode.cn/problems/longest-palindromic-substring/
暴力 超时
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22public String longestPalindrome(String s) {
String result = "";
for (int i=0; i<s.length(); i++) {
for (int j=i+1; j<=s.length(); j++) {
String str = s.substring(i, j);
if (this.isSame(str) && str.length()>result.length()) {
result = str;
}
}
}
return result;
}
private boolean isSame(String s) {
int mid = (s.length() - 1) / 2;
for (int i=0; i<=mid; i++) {
if (s.charAt(i) != s.charAt(s.length()-1-i)) {
return false;
}
}
return true;
}动态规划
1
2
3
4
5dp[i][j] [i, j]是否是回文子串
s[i] != s[j] dp[i][j] = false
s[i] == s[j]
j-i<=1 true
j-i>1 dp[i+1][j-1]
1 | public String longestPalindrome(String s) { |
编辑距离
判断子序列
https://leetcode.cn/problems/is-subsequence/
- 回溯搜索 超时
1
2
3
4
5
6
7
8
9
10
11
12
13private Set<String> set = new HashSet<>();
public boolean isSubsequence(String s, String t) {
this.backtrack(t, 0, "");
return this.set.contains(s);
}
// 不同的深度含有的元素个数不同,可以看到,不同个数的元素都包含进来了,找到了所有的子集,即子序列
private void backtrack(String s, int startIndex, String path) {
this.set.add(path);
for (int i=startIndex; i<s.length(); i++) {
backtrack(s, i+1, path+s.charAt(i));
}
} - dp
下面的递推公式是为了删除操作,其实可以当作公共子序列长度问题做
1 | 1. dp[i][j] 含义以s[i-1]结尾的字符串 与 以 t[j-1]结尾的字符串含有的相同子序列长度 |
1 | public boolean isSubsequence(String s, String t) { |
- 纠正dp的含义
上面的方法递推公式只删除 t的字符,导致有些时候dp含义无效,如果递推公式改为
dp[i][j] = Math.max(dp[i-1][j], dp[i][j-1]) 两个都可能删除,dp意义就正确了
1 | public boolean isSubsequence(String s, String t) { |
- 双指针,更好理解
1
2
3
4
5
6
7
8
9
10
11
12
13public boolean isSubsequence(String s, String t) {
// 双指针,遍历s,在t中逐个寻找
int indexS=0, indexT=0;
while (indexS<s.length() && indexT<t.length()) {
if (s.charAt(indexS) == t.charAt(indexT)) {
indexS++;
indexT++;
} else {
indexT++;
}
}
return indexS == s.length();
}
不同的子序列
https://leetcode.cn/problems/distinct-subsequences/
1 | 1. dp[i][j] 以s[i-1]结尾的字符串的子串中 t的[0, j-1]字符串出现的次数 |
1 | public int numDistinct(String s, String t) { |
两个字符的删除操作
https://leetcode.cn/problems/delete-operation-for-two-strings/
1 | 1. dp[i][j] word1[0, i-1] word2[0, j-1] 相等时需要的最少删除次数 |
1 | public int minDistance(String word1, String word2) { |
编辑距离
https://leetcode.cn/problems/edit-distance/
1 | 1. dp[i][j] word1[0, i-1] 与 子串word2[0, j-1]最短的编辑距离 |
1 | public int minDistance(String word1, String word2) { |
子序列(不连续)
最长上升子序列
https://leetcode.cn/problems/longest-increasing-subsequence/
注意dp数组的含义,必须以nums[i]结尾,也就是序列中必须包含nums[i],且以它结尾
1 | 1. dp[i] 以nums[i]结尾的子序列最长长度 |
1 | public int lengthOfLIS(int[] nums) { |
最长公共子序列
https://leetcode.cn/problems/longest-common-subsequence/
1 | 1. dp[i][j] text1[0, i-1] text2[0, j-1] 最长公共子序列的长度,没有限制必须以text1[i-1] text2[j-1]结尾 |
1 | public int longestCommonSubsequence(String text1, String text2) { |
不相交的线
https://leetcode.cn/problems/uncrossed-lines/
1 | 其实就是求最长公共子序列的长度 |
1 | public int maxUncrossedLines(int[] nums1, int[] nums2) { |
子序列(连续)
最长连续递增子序列
https://leetcode.cn/problems/longest-continuous-increasing-subsequence/
- dp
注意dp数组的含义,必须以nums[i]结尾,也就是序列中必须包含nums[i],且以它结尾
1 | 1. dp[i] 以i为结尾的最长连续子序列长度 |
1 | public int findLengthOfLCIS(int[] nums) { |
- 普通遍历
1
2
3
4
5
6
7
8
9
10
11
12
13
14public int findLengthOfLCIS(int[] nums) {
int length = nums.length;
int result = 1;
int temp = 1;
for (int i=1; i<length; i++) {
if (nums[i] > nums[i-1]) {
temp++;
} else {
result = Math.max(temp, result);
temp = 1;
}
}
return Math.max(temp, result);
} - 如果需要输出最长的子序列
使用left 与 right 保存临时长度与最初的长度
1 | public int findLengthOfLCIS(int[] nums) { |
最长重复子数组
https://leetcode.cn/problems/maximum-length-of-repeated-subarray/
包括nums1[i-1],且以nums1[i-1]结尾的子数组
包括nums2[j-1],且以nums2[j-1]结尾的子数组
这两个子数组如果相同,dp[i][j]就是相同子数组最大的长度
如果不相同,就是dp[i][j]就是0
1 | 连续数组 |
1 | public int findLength(int[] nums1, int[] nums2) { |
最大子数组和
https://leetcode.cn/problems/maximum-subarray/
- dp
1
2
3
4
5
6
7
81. dp[i] 以nums[i] 结尾的连续数组,最大和
2. dp[i] = Math.max(dp[i-1]+nums[i], nums[i])
nums[i] 结尾子数组的和,只有两种情况,包括前面的,还有不包括前面的
3. dp[0] = nums[0]
4. 左右
5. [-2, 1, -3, 4, -1, 2, 1, -5, 4]
-2 1 -2 4 3 5 6 1 5
遍历dp返回最大
1 | public int maxSubArray(int[] nums) { |
- 贪心
自己写的屎代码
1 | // 如果nums[i] 加上的前面连续序列和是负数,肯定还不如不加, |
别人的代码
1 | // 如果nums[i] 加上的前面连续序列和是负数,肯定还不如不加, |