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

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

动规五部曲:

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

2.确定递推公式:关键在于判断s[i]与s[j]是否相同。

(1)如果s[i]与s[j]相同:那么dp[i][j] = dp[i + 1][j - 1] + 2,如下图所示。

(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]),如下所示。

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相同时递推公式dp[i][j] = dp[i + 1][j - 1] + 2可以看出,递推公式计算不到i和j相同时候的情况,因此需要提前初始化一下:当i和j相同的时候,dp[i][j]一定是等于1的,即:一个字符的回文子序列就是它本身。

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

4.确定遍历顺序:从递推公式可以看出,dp[i][j]依赖于dp[i + 1][j - 1],dp[i + 1][j]和dp[i][j - 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];
    }
}
相关推荐
夜思红尘8 小时前
算法--双指针
python·算法·剪枝
散峰而望8 小时前
【算法竞赛】C++函数详解:从定义、调用到高级用法
c语言·开发语言·数据结构·c++·算法·github
CoderCodingNo8 小时前
【GESP】C++五级真题(贪心思想考点) luogu-B4071 [GESP202412 五级] 武器强化
开发语言·c++·算法
我有一些感想……8 小时前
An abstract way to solve Luogu P1001
c++·算法·ai·洛谷·mlp
前端小L8 小时前
双指针专题(三):去重的艺术——「三数之和」
javascript·算法·双指针与滑动窗口
智者知已应修善业9 小时前
【求等差数列个数/无序获取最大最小次大次小】2024-3-8
c语言·c++·经验分享·笔记·算法
LYFlied10 小时前
【每日算法】LeetCode 416. 分割等和子集(动态规划)
数据结构·算法·leetcode·职场和发展·动态规划
多米Domi01110 小时前
0x3f 第19天 javase黑马81-87 ,三更1-23 hot100子串
python·算法·leetcode·散列表
历程里程碑10 小时前
滑动窗口最大值:单调队列高效解法
数据结构·算法·leetcode
課代表11 小时前
从初等数学到高等数学
算法·微积分·函数·极限·导数·积分·方程