【算法 - 动态规划】最长回文子序列

上篇文章中,我们学习一个新的模型: 样本对应模型,该模型的套路就是:以结尾位置为出发点,思考两个样本的结尾都会产生哪些可能性

而前篇文章中的 纸牌博弈问题 属于 [L , R]上范围尝试模型。该模型给定一个范围,在该范围上进行尝试,套路就是 思考 [L ,R] 两端该如何取舍。

本篇文章我们通过一道中等难度的力扣题,再来熟悉 范围尝试模型 的套路,注意与 上篇文章的最长公共子序列 作对比哦!

力扣516. 最长回文子序列

给你一个字符串 s ,找出其中最长的回文子序列,并返回该序列的长度。

子序列定义为:不改变剩余字符顺序的情况下,删除某些字符或者不删除任何字符形成的一个序列。

示例 1:

输入: s = “bbbab”

输出: 4

解释: 一个可能的最长回文子序列为 “bbbb” 。

示例 2:

输入: s = “cbbd”

输出: 2

解释: 一个可能的最长回文子序列为 “bb” 。

学会了 上篇最长公共子序列 之后,这道题目就有一个讨巧的方法:

若翻转输入的字符串,那么原本的字符串与翻转后的字符串的 最长公共子序列 就是 最长回文子序列

只需稍加修改,就能套用上面的代码了!

代码

public static int palindromeSubsequence(String s) {
    char[] s1 = s.toCharArray();
    int N = s1.length;
    char[] s2 = new char[N];
    // 翻转 s 字符串
    for (int i = 0; i < N; i++) {
        s2[N - i - 1] = s1[i];
    }
    // 调用 判断最长公共子序列 的动态规划函数
    return longestCommonSubsequence(s1,s2);
}

// 该函数是 上篇文章末尾 的 动态规划版 
public static int longestCommonSubsequence(char[] str1, char[] str2) {
    int N = str1.length;
    int M = str2.length;
    ...
}

除了上面讨巧的办法外,我们依然采用最朴素的 暴力递归 来思考这道题目。

递归的准备

定义递归函数的功能: 返回 str 中 [L ... R] 范围上字符串的最长回文子序列。

思考递归需要的参数: str 字符串,两端范围 L, R。

明确递归的边界条件:

  • 当字符串长度为 1 即 L == R 时,找到了一个长度为 1 的回文序列,返回 1 。
  • 当字符串长度为 2 即 L == R - 1 时,若两个字符一致,即找到了一个长度为 2 的回文序列,返回 2 。否则返回 1。

思路

这道题就是典型的 范围尝试模型 ,因此,递归就可以按照 开头和结尾两端都会产生哪些可能性 的思路来划分情况:

  • 回文子序列 既不以 L 结尾,也不以 R 结尾;
  • 回文子序列 L 结尾,不以 R 结尾;
  • 回文子序列 不以 L 结尾, R 结尾;
  • 回文子序列 既以 L 结尾,也以 R 结尾。

因为要求 最长 回文子序列,因此要返回这四种情况当中的最大值。

代码

public static int palindromeSubsequence(String s) {
    if (s == null || s.length() == 0) {
        return 0;
    }
    char[] str = s.toCharArray();
    return process(str, 0, str.length - 1);
}

public static int process(char[] str, int L, int R) {
    if (L == R) {
        return 1;
    }
    if (L == R - 1) {
        return str[L] == str[R] ? 2 : 1;
    }
    int p1 = process(str, L + 1, R - 1);
    int p2 = process(str, L, R - 1);
    int p3 = process(str, L + 1, R);
    int p4 = str[L] != str[R] ? 0 : (2 + process(str, L + 1, R - 1));
    return Math.max(Math.max(p1, p2), Math.max(p3, p4));
}

代码解释

注意: 第四种情况 p4 中,如果直接调用 process(str, L, R),则会产生死循环。为使递归正常运行,要将都以 LR 结尾单独拿出来判断,若相等,去掉两端相等的字符再进行递归调用,回文子序列的长度自然要加上两端 2 的长度。


下面我们通过画 dp 表,探寻该递归如何转化为更加优化的动态规划。

以 str = “abcb1a” 为例。

可变的参数有两个,判断长度范围的 LR。因此,需要设置一个二维的 dp 表数组,由于 L, R 的取值范围从 0 开始到字符串长度减一,因此数组大小设置为 dp[N][N]

根据递归函数中代码逻辑发现:

  1. 当字符串中仅剩一个字符时,回文长度为 1 ,其余均为 0。
    if (L == R) {
        return 1;
    }

因此 dp 数组对角线上的数值均为 1 。

  1. 当字符串长度为 2 ,若两个字符一致,即找到了一个长度为 2 的回文序列,返回 2 。否则返回 1。
    if (L == R - 1) {
        return str[L] == str[R] ? 2 : 1;
    }

  1. 普遍情况下,依赖 左,下,左下 三个地方的 最大值
    int p1 = process(str, L + 1, R - 1);
    int p2 = process(str, L, R - 1);
    int p3 = process(str, L + 1, R);
    int p4 = str[L] != str[R] ? 0 : (2 + process(str, L + 1, R - 1));
    return Math.max(Math.max(p1, p2), Math.max(p3, p4));


  1. 范围尝试模型的 dp 表最大特点是左下角无效。递归函数调用的是 process(str, 0, str.length - 1),因此最终答案应该取 dp[0][N-1]== 5。

动态规划代码

public static int palindromeSubsequence(String s) {
    if (s == null || s.length() == 0) {
        return 0;
    }
    if (s.length() == 1) {
        return 1;
    }
    char[] str = s.toCharArray();
    int N = str.length;
    int[][] dp = new int[N][N];
    dp[N - 1][N - 1] = 1;
    // 单独填写对角线和相邻斜线的值
    for (int i = 0; i < N - 1; i++) {
        dp[i][i] = 1;
        dp[i][i + 1] = str[i] == str[i + 1] ? 2 : 1;
    }
    // 从下往上 从左往右 找最大值填写
    // 递归中的 p1 情况不需要考虑了
    for (int i = N - 3; i >= 0; i--) {
        for (int j = i + 2; j < N; j++) {
            dp[i][j] = Math.max(dp[i][j - 1], dp[i + 1][j]);
            if (str[i] == str[j]) {
                dp[i][j] = Math.max(dp[i][j], dp[i + 1][j - 1] + 2);
            }
        }
    }
    return dp[0][N - 1];
}

代码解释

一图胜千言:

在递归版本中,红色分别依赖紫、蓝、黄,并 取最大值

    int p1 = process(str, L + 1, R - 1);
    int p2 = process(str, L, R - 1);
    int p3 = process(str, L + 1, R);
    int p4 = str[L] != str[R] ? 0 : (2 + process(str, L + 1, R - 1));
    return Math.max(Math.max(p1, p2), Math.max(p3, p4));

思考一下,紫色部分的值是怎么来的?它也同样依赖了左,下,左下 三个地方的 最大值。因此,紫色 部分一定不小于 蓝色 部分,同理 黄色 部分也一定不小于 蓝色 部分。
因为要求最大值,因此可以忽略掉蓝色部分 p1 的递归调用,只需考虑 p2, p3 的调用。

    dp[i][j] = Math.max(dp[i][j - 1], dp[i + 1][j]);
    if (str[i] == str[j]) {
        dp[i][j] = Math.max(dp[i][j], dp[i + 1][j - 1] + 2);
    }

因此在动态规划循环中排除了蓝色部分的情况,先求出紫色与黄色的最大值,只有当 str[L] == str[R] 时,再与蓝色部分比出最大值。

这就是使用 严格表依赖 的好处 —— 可以 做进一步的优化

总结

上篇文章和本文分别讲解了 最长公共子序列问题最长回文子序列问题 。看似很相似的题目,实际使用了两种不同的模型:样本对应模型范围尝试模型 。根据不同模型的套路相信小伙伴也能够轻松应对类似的题目了!

下篇文章我们将介绍一个综合性非常高的题目,并使用一个新的模型来应对,敬请期待吧~

~ 点赞 ~ 关注 ~ 不迷路 ~!!!

------------- 往期回顾 -------------
【算法 - 动态规划】最长公共子序列问题
【算法 - 动态规划】力扣 691. 贴纸拼词
【算法 - 动态规划】原来写出动态规划如此简单!
【算法 - 动态规划】从零开始学动态规划!(总纲)
【堆 - 专题】“加强堆” 解决 TopK 问题!
AC 此题,链表无敌!!!

  • 27
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值