【算法 - 动态规划】最长回文子序列_最长回文子序列动态规划-程序员宅基地

技术标签: 算法  动态规划  

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

而前篇文章中的 纸牌博弈问题 属于 [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 此题,链表无敌!!!

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_46879314/article/details/136213302

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签