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

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

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/401066.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【洛谷 P8780】[蓝桥杯 2022 省 B] 刷题统计 题解(二分查找+数学)

[蓝桥杯 2022 省 B] 刷题统计 题目描述 小明决定从下周一开始努力刷题准备蓝桥杯竞赛。他计划周一至周五每天做 a a a 道题目&#xff0c;周六和周日每天做 b b b 道题目。请你帮小明计算&#xff0c;按照计划他将在第几天实现做题数大于等于 n n n 题? 输入格式 输入一…

Android14 InputManager-InputReader的处理

IMS启动时会调用InputReader.start()方法 InputReader.cpp status_t InputReader::start() {if (mThread) {return ALREADY_EXISTS;}mThread std::make_unique<InputThread>("InputReader", [this]() { loopOnce(); }, [this]() { mEventHub->wake(); });…

Nignx的搭建与核心配置

目录 一、Nginx是什么&#xff1f; 1、Nginx概述 2、Nginx模块与作用 3、Nginx三大作用&#xff1a;反向代理&#xff0c;负载均衡&#xff0c;动静分离 nginx七层负载均衡调度算法&#xff08;六种&#xff09; 1、轮询&#xff08;默认调度算法&#xff09; 2、加权轮…

雨量水位在线监测站

TH-SW2随着科技的不断进步&#xff0c;雨量水位在线监测站在防汛、水资源管理、环境保护等领域发挥着越来越重要的作用。传统的雨量水位监测方法往往存在精度不高、实时性不强等问题&#xff0c;而雷达式测量技术的出现&#xff0c;则为这一领域带来了革命性的变化。 一、雨量水…

【踩坑专栏】主机ping虚拟机失败

我出现的问题finalshell连接超时&#xff0c;ping了一下发现ping都ping不通&#xff0c;于是发现问题所在。 最开始我是把虚拟机的网络设置改为桥接模式&#xff0c;问题解决了&#xff0c;但是这种模式的问题就是每次开机&#xff0c;ip都会改变&#xff0c;因此非常麻烦&…

零到大师:嵌入式Linux学习书单分享

大家好&#xff0c;我是知微&#xff01; 上一篇推荐的书单嵌入式软件必读10本书_单片机篇&#xff0c;收到反响很好。再推荐一篇嵌入式Linux相关的书单。 《鸟哥的Linux私房菜》 鸟哥的Linux系列适合零基础小伙伴&#xff0c;从电脑基础到文件系统、shell脚本等等&#xff…

【PX4学习笔记】04.QGC地面站的使用

目录 文章目录 目录PX4代码烧入PX4固件代码的烧入方式1PX4固件代码的烧入方式2 QGC地面站的基础使用连接地面站的方式查看关键的硬件信息 QGC地面站的Application Settings模块Application Settings模块-常规界面单位其他设置数据持久化飞机中的数传日志飞行视图计划视图自动连…

Unity Shader ASE基础效果思路与代码(一):遮罩、硬边溶解、光边溶解、UV扰动

Unity Shader ASE基础效果思路与代码(一)&#xff1a;遮罩、硬边溶解、光边溶解、UV扰动 文章目录 Unity Shader ASE基础效果思路与代码(一)&#xff1a;遮罩、硬边溶解、光边溶解、UV扰动遮罩效果硬边溶解光边溶解UV扰动 遮罩效果 效果展示&#xff1a; 思路与代码&#xff1…

SQL注入:堆叠注入-强网杯[随便注]

目录 什么是堆叠注入&#xff1f; 强网杯-随便注 rename && alter绕过 prepare绕过 Handle绕过 靶机&#xff1a;BUUCTF在线评测 什么是堆叠注入&#xff1f; 在一些场景中&#xff0c;应用程序支持一次执行多条SQL语句&#xff0c;我们称为堆叠查询&#xff0c;…

HTML5-CSS3

一、HTML5的新特性 HTML5 的新增特性主要是针对于以前的不足&#xff0c;增加了一些新的标签、新的表单和新的表单属性等。 这些新特性都有兼容性问题&#xff0c;基本是 **IE9 以上版本的浏览器**才支持&#xff0c;如果不考虑兼容性问题&#xff0c;可以大量使用这些新特性…

前端是做什么的?

前端很多时候代表从事前端开发的工程师&#xff0c;定义上来说&#xff0c;他们负责规划、设计、构建和运行网站、软件程序和基于 Web 的应用程序的用户界面系统。通俗点来说&#xff0c;我们看到的网站、点击网站所有的按钮以及和网站的交互都是前端所进行的工作。 概述 前端…

pclpy 窗口可视化多个点云

pclpy 窗口可视化多个点云 一、算法原理二、代码三、结果1.可视化结果 四、相关数据五、问题与解决方案1.问题2.解决 一、算法原理 原理看一下代码写的很仔细的。。目前在同一个窗口最多可视化两个点云。。 二、代码 from pclpy import pcldef CloudShow(cloud1, cloud2):&q…

java数据结构与算法刷题-----LeetCode222. 完全二叉树的节点个数

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 1. 法一&#xff1a;利用完全二叉树性质&#xff0c;进行递归二分查找 解…

145. Binary Tree Postorder Traversal(二叉树的后序遍历)

问题描述 给你一棵二叉树的根节点 root &#xff0c;返回其节点值的 后序遍历 。 问题分析 此问题与二叉树的前序遍历分析类似&#xff0c;只是在数组合并的时候顺序发生一下变化就行了。 代码 int* postorderTraversal(struct TreeNode* root, int* returnSize) {if(root…

【电子书】计算机课程

资料 wx&#xff1a;1945423050 个人整理了一些互联网电子书 计算机课程 Netty权威指南&#xff08;第2版&#xff09;.epubSharePoint Server 2016 IT Pro 部署指南.epubTensorFlow自然语言处理.epubWebGIS之OpenLayers全面解析.epub从Paxos到Zookeeper分布式一致性原理与实践…

《图解设计模式》笔记(一)适应设计模式

图灵社区 - 图解设计模式 - 随书下载 评论区 雨帆 2017-01-11 16:14:04 对于设计模式&#xff0c;我个人认为&#xff0c;其实代码和设计原则才是最好的老师。理解了 SOLID&#xff0c;如何 SOLID&#xff0c;自然而然地就用起来设计模式了。Github 上有一个 tdd-training&…

Javascript中var和let之间的区别

文章目录 一.变量提升(声)二.let和var的区别 区别&#xff1a; 1、var有变量提升&#xff0c;而let没有&#xff1b; 2、let不允许在相同的作用域下重复声明&#xff0c;而var允许&#xff1b; 3、let没有暂时性死区问题&#xff1b; 4、let创建的全局变量没有给window设置对应…

【广度优先搜索】【网格】【割点】【 推荐】1263. 推箱子

作者推荐 视频算法专题 涉及知识点 广度优先搜索 网格 割点 并集查找 LeetCode:1263. 推箱子 「推箱子」是一款风靡全球的益智小游戏&#xff0c;玩家需要将箱子推到仓库中的目标位置。 游戏地图用大小为 m x n 的网格 grid 表示&#xff0c;其中每个元素可以是墙、地板或…

ctfshow web入门 web141-145

1.web141 ^\w$表示在开头和末尾匹配字母数字_&#xff0c;传入的v3值不能有字母数字_&#xff0c;即无字母的命令执行 php中1-phpinfo()是可以执行的&#xff0c;加减乘除都可以实现 这里或&#xff0c;异或&#xff0c;取反等运算都可以 这里采用羽师傅的异或脚本生成paylo…

小白水平理解面试经典题目LeetCode 404 Sum of Left Leaves【Tree】

404 左叶之和 小白翻译 给定二叉树的root&#xff0c;返回所有左叶的总和。 叶子是没有子节点的节点。左叶是另一个节点的左子节点的叶。 例子 小白教室做题 在大学某个自习的下午&#xff0c;小白坐在教室看到这道题。想想自己曾经和白月光做题&#xff0c;现在大过年的&a…