- dp[i][j] 表示 字符串 [i , j]范围内的最长回文子序列长度
- 递推公式:
当 s[i] == s[j] 时,如果 i 和 j 相等,则回文子序列长度为1
当 s[i] == s[j] 时当 i 和 j不相等时,dp[i][j] = dp[i + 1][j - 1] + 2;
当s[i] != s[j],那么说明s[i]和s[j]的同时加入 并不能增加[i,j]区间回文子序列的长度,那么分别加入s[i]、s[j]看看哪一个可以组成最长的回文子序列。
加入s[j]的回文子序列长度为dp[i + 1][j]。
加入s[i]的回文子序列长度为dp[i][j - 1]。
那么dp[i][j]一定是取最大的,即:dp[i][j] = max(dp[i + 1][j], dp[i][j - 1]); - 初始化 默认全部回文子序列长度为0
- dp[i][j] 依赖于 dp[i + 1][j - 1] ,所以遍历i的时候一定要从下到上遍历,这样才能保证下一行的数据是经过计算的。
j的话,可以正常从左向右遍历。
对回文串类型的题,似乎外循环都需要从下到上进行遍历
class Solution {
public:
int longestPalindromeSubseq(string s) {
vector<vector<int>> dp(s.size(), vector<int>(s.size(), 0));
for(int i = s.size() - 1; i >= 0; i--){
for(int j = i; j < s.size(); j++){
if(s[i] == s[j] && j == i){
dp[i][j] = 1;
}else if(s[i] == s[j]){
dp[i][j] = dp[i + 1][j - 1] + 2;
}else{
dp[i][j] = max(dp[i + 1][j], dp[i][j - 1]);
}
}
}
return dp[0][s.size() - 1];
}
};