代码随想录 516.最长回文子序列

思路:本题是求回文子序列,而不是回文子串。回文子串要求一定连续,但回文子序列可以不连续。

动规五部曲:

1.确定dp数组(dp table)及其下标的含义:dpij表示字符串s在i,j范围内最长的回文子序列的长度为dpij

2.确定递推公式:关键在于判断si与sj是否相同。

(1)如果si与sj相同:那么dpij = dpi + 1j - 1 + 2,如下图所示。

(2)如果si与sj不同:说明si与sj的同时加入不能增加i,j区间回文子序列的长度,那么分别加入si、sj看看哪一个可以组成最长的回文子序列。

------加入sj的回文子序列的长度为:dpi + 1j

------加入si的回文子序列的长度为:dpij - 1

dpij一定是取最大的,那么dpij = max(dpi + 1j,dpij - 1),如下所示。

cpp 复制代码
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]);
}

3.dp数组如何初始化:

(1)首先考虑i和j相同的情况,从i和j相同时递推公式dpij = dpi + 1j - 1 + 2可以看出,递推公式计算不到i和j相同时候的情况,因此需要提前初始化一下:当i和j相同的时候,dpij一定是等于1的,即:一个字符的回文子序列就是它本身。

(2)其他情况:dpij初始化为0就行。这样递推公式dpij = max(dpi + 1j,dpij - 1)中,dpij才不会被初始值所覆盖。

4.确定遍历顺序:从递推公式可以看出,dpij依赖于dpi + 1j - 1,dpi + 1j和dpij - 1,如下图所示。因此遍历顺序为从下到上,从左到右

5.举例推导dp数组。以输入s: "cbdd"为例,dp数组的状态如下图所示。

附代码:

java 复制代码
class Solution {
    public int longestPalindromeSubseq(String s) {
        int len = s.length();
        int[][] dp = new int[len + 1][len + 1];
        for(int i = len - 1;i >= 0;i--){
            dp[i][i] = 1;
            //已经初始化完i和j相等的情况(此时子序列长度为1),因此判断子序列长度只需在i + 1位开始
            for(int j = i + 1;j < len;j++){
                if(s.charAt(i) == s.charAt(j)){
                    dp[i][j] = dp[i + 1][j - 1] + 2;
                }else{
                    dp[i][j] = Math.max(dp[i + 1][j],dp[i][j - 1]);
                }
            }
        }
        return dp[0][len - 1];
    }
}
相关推荐
JAVA面经实录9173 小时前
Java 数据结构与算法 (终极完整学习文档)
java·数据结构·算法
开源Z4 小时前
LeetCode 42 · 接雨水:从暴力到双指针的三步优化
算法·leetcode
旖-旎5 小时前
《LeetCode 695 岛屿的最大面积 FloodFill DFS 解法》
c++·算法·力扣·深度优先遍历·floodfill
syagain_zsx5 小时前
STL 之 vector 讲练结合
c++·算法
MartinYeung57 小时前
[论文学习]DP2Unlearning:高效且具保证的大型语言模型遗忘框架(基于差分隐私的 LLM Unlearning 方法)
学习·算法·语言模型
Tian_Hang7 小时前
C++原型模式(Protype)
开发语言·c++·算法
bIo7lyA8v7 小时前
算法复杂度的渐进分析与实际运行时间的差异的技术8
算法
yuan199978 小时前
欧拉梁静力与屈曲计算的 MATLAB 实现(有限差分法 + 解析解)
开发语言·算法·matlab
汉克老师9 小时前
GESP7级C++考试语法知识(二、指数函数(3、综合练习)
c++·算法·数学建模·指数函数·gesp7级·复利