暴力递归转动态规划(七)

news2025/1/11 22:53:34

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

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

示例 1:

输入:s = “bbbab”
输出:4
解释:一个可能的最长回文子序列为 “bbbb” 。
示例 2:

输入:s = “cbbd”
输出:2
解释:一个可能的最长回文子序列为 “bb” 。

换句话说,就是删减完的字符串读出来左右两边是一样的:
如字符串 String s = “as1d2jk3a3jkh2gv1d”,最长回文子序列就是123a321。

这道题解法可参考上篇文章最长公共子序列。
将给定的字符串s进行逆序,生成新的字符串s1,求字符串s和字符串s1的最长公共子序列的长度即为最长回文子序列的长度。
举例: String s = “as1d2jk3a3jkh2gv1d” String s1 = “d1vg2hkj3a3kj2d1sa”。
公共子序列为:123a321,即为所求的最长回文子序列。

代码:

public static int lpsl1(String str) {
        if (null == str || str.length() == 0) {
            return 0;
        }

        char[] str1 = str.toCharArray();
        char[] str2 = reverse(str1);

        return process(str1, str2, str1.length - 1, str2.length - 1);
    }

    public static char[] reverse(char[] str) {
        int N = str.length;
        char[] str1 = new char[N];

        for (int i = 0; i < str.length; i++) {
            str1[i] = str[--N];
        }
        return str1;
    }

    public static int process(char[] str1, char[] str2, int i, int j) {
        if (i == 0 && j == 0) {
            return str1[i] == str2[j] ? 1 : 0;
        } else if (i == 0) {
            if (str1[i] == str2[j]) {
                return 1;
            } else {
                return process(str1, str2, i, j - 1);
            }
        } else if (j == 0) {
            if (str1[i] == str2[j]) {
                return 1;
            } else {
                return process(str1, str2, i - 1, j);
            }
        } else {
            int p1 = process(str1, str2, i - 1, j);
            int p2 = process(str1, str2, i, j - 1);
            int p3 = str1[i] == str2[j] ? 1 + process(str1, str2, i - 1, j - 1) : 0;

            return Math.max(p3, Math.max(p1, p2));
        }
    }

刚才的方法是根据公共子序列推演出来的方法,那现在让我们来直面这个问题,同样是采用暴力递归+动态规划的方式来进行解答。

暴力递归
依然是从暴力递归开始,写出暴力递归方法后优化成动态规划。
暴力递归方法主要是返回str[L…R]范围内的最大回文子序列。
共有以下几种情况:

  1. base case:让L = R时,说明只剩一个字符,每个字符的本身都算是自己的回文子序列,所以return 1。
    1.1 如果L = R - 1时,还L 到 R 位置剩两个字符,如果这两个字符相等,说明回文子序列的长度为2,否则每个字符都是自己单独的回文子序列,返回1。

base case考虑完之后,就剩下其余的共4种情况:

  1. 回文子序列可能不以L位置结尾,但是必须以R位置结尾。
  2. 回文子序列可能不以R位置结尾,但是必须以L位置结尾。
  3. 回文子序列即不以L位置结尾,也不以R位置结尾。
  4. 回文子序列必须以L位置结尾 && 必须以R位置结尾。

根据这几种情况,来写递归方法。
注:下面的递归方法在LC中跑,会因为超时不通过,但是通过给的例子可以发现,不是逻辑性错误,是因为超时。后续将暴力递归优化成动态规划就好了。

代码

public static int lps2(String s) {
        if (null == s || s.length() == 0) {
            return 0;
        }

        if (s.length() == 1) {
            return 1;
        }
        char[] str = s.toCharArray();
        return process2(str,0,str.length - 1);
    }
	//递归方法返回 str[L...R]范围内的最大回文子序列
    public static int process2(char[] str, int L, int R) {
        if (L == R) {
            return 1;
        } else if (L == R - 1) {
            return str[L] == str[R] ? 2 : 1;
        } else {

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

动态规划
依然是根据上面暴力递归的代码改动态规划,根据代码可发现,可变参数是L 和 R,变化范围是 0 ~ str.lengtth - 1。所以可以确定dp表的大小。

代码
从暴力递归优化成动态规划的第一版代码,代码中有一些小技巧。
比如说根据base case给dp表赋值时,先将dp[N - 1][N - 1]赋值为 -1 (根据base case L = R得知),再用第一个循环将 L = R 和 L = L + 1 赋值。(先赋值dp[N - 1][ N - 1]能够确保不越界)
第二个循 L = N - 3开始 R 从 L + 2开始,是因为第一个循环已经将dp表的最后两行填满了,所以 L 从 N - 3开始填充,R 从 L + 2位置开始。
在根据递归代码可以看出,递归中调用process(L + 1,R ) ,process(L,R - 1) 和 process(L + 1,R - 1)可以看出调用的依赖关系是左、下和左下三个位置。所以整个填充顺序如图所示:
在这里插入图片描述

   if (null == s || 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;
        }


        for (int L = N - 3; L >= 0; L--) {
            for (int R = L + 2; R < N; R++) {
                int p1 = dp[L + 1][R];
                int p2 = dp[L][R - 1];
                int p3 = dp[L + 1][R - 1];
                int p4 = str[L] == str[R] ? 2 + dp[L + 1][R - 1] : 0;

                dp[L][R] = Math.max(Math.max(p1, p2), Math.max(p3, p4));
            }
        }
        return dp[0][N - 1];
    }

再次优化
根据依赖位置(左、下和左下)可以看出,左下的位置是没有必要的,因为求的是最大的回文子序列,左下是( L + 1, R - 1)的位置,它的范围一定比(L + 1,R) 和 (L , R - 1)小,所以不需要它,那么只需要p1,p2和p4进行比较即可,而p4只有在str[L] == str[R]时,才参与比较,所以还可以进一步优化。

代码

public static int dp2(String s) {
        if (null == s || 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;
        }


        for (int L = N - 3; L >= 0; L--) {
            for (int R = L + 2; R < N; R++) {
                dp[L][R] = Math.max(dp[L + 1][R],dp[L][R - 1]);
                if (str[L] == str[R]){
                    dp[L][R] = Math.max((dp[L][R]),(2 + dp[L + 1][R - 1]));
                }
            }
        }
        return dp[0][N - 1];
    }

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【【萌新的FPGA学习之按键控制LED实验】】

按键控制LED实验 在写这篇文章之前我必须对我的错误表示深刻的道歉 因为我之前的文章自己也是边看边学给大家带来了大的困扰 抱歉抱歉 我们这里讲述一下综合和仿真的关系 其实我们更多的是应该关注仿真下得到的波形情况 然后分析 对于综合&#xff0c;综合的最大的目的还是看功…

计算机等级考试—信息安全三级真题二

目录 一、单选题 二、填空题 三、综合题 一、单选题

数据结构的奇妙世界:实用算法与实际应用

文章目录 数据结构和算法的基本概念数据结构数组链表栈队列树图 算法 常见的数据结构和算法排序算法快速排序示例 数据结构的应用数据库管理系统图像处理网络路由 数据结构和算法的性能分析时间复杂度空间复杂度 如何更好地编写代码避免常见错误结论 &#x1f389;欢迎来到数据…

Qt地铁智慧换乘系统浅学( 三 )最少路径和最少换乘实现

本算法全都基于广度优先 概念最短路径实现所用容器算法思路 最少换乘实现所需容器算法思路 成果展示代码实现判断是最短路径还是最少换乘最短路径代码实现最少换乘代码实现根据所得List画出线路 ui界面的维护&#xff08;前提条件&#xff09;界面初始化combox控件建立槽函数 概…

把Eclipse整个文件夹添加到Microsoft Defender的排除项中

一.原因&#xff1a; Windows 10卫士显著降低了Eclipse的速度&#xff0c;原因是Windows 10卫士扫描JAR文件。这个问题已经报告给微软了。在此之前&#xff0c;解决此问题的一个方法是将Eclipse根目录添加到Windows 10 Defender的排除列表中&#xff0c;详细步骤在这里共享。 …

前端JavaScript入门到精通,javascript核心进阶ES6语法、API、js高级等基础知识和实战 —— JS基础(五)

接受自己原本的样子&#xff0c; 比努力扮演另一个轻松多了。 思维导图 对象 什么是对象 对象使用 遍历对象 索引号是字符串型&#xff0c;不推荐遍历数组。 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><me…

docker实现mysql主从复制(巨详细!!!)

docker实现mysql主从复制&#xff08;巨详细&#xff01;&#xff01;&#xff01;&#xff09; 新建主机服务容器实例3307进入/mydata/mysql-master/conf目录下新建my.cnf修改完配置后重启master实例进入mysql-master容器master容器实例内创建数据同步用户新建 服务器容器实例…

【论文阅读】内存数据库并发控制算法的实验研究

内存数据库并发控制算法的实验研究 原文链接jos.org.cn/jos/article/pdf/6454 摘要 并发控制算法的基本思想归纳为"先定序后检验”&#xff0c;基于该思想对现有各类并发控制算法进行 了重新描述和分类总结&#xff0c;于在开源内存型分布式事务测试床 3TS 上的实际对比实…

Tune-A-Video论文阅读

论文链接&#xff1a;Tune-A-Video: One-Shot Tuning of Image Diffusion Models for Text-to-Video Generation 文章目录 摘要引言相关工作文生图扩散模型文本到视频生成模型文本驱动的视频编辑从单个视频生成 方法前提DDPMsLDMs 网络膨胀微调和推理模型微调基于DDIM inversio…

动手学深度学习(pytorch版)第二章-2.3线性代数Note-linear-algebra

类型 标量&#xff1a;仅包含一个数值被称为标量 向量&#xff1a;向量可以被视为标量值组成的列表 矩阵&#xff1a;正如向量将标量从零阶推广到一阶&#xff0c;矩阵将向量从一阶推广到二阶。 A torch.arange(20).reshape(5, 4) A.T //转置 张量&#xff1a;是描述具有…

[36c3 2019]includer

[36c3 2019]includer 题目描述&#xff1a;Just sitting here and waiting for PHP 8.0 (lolphp). 首先来了解一下临时文件包含之PHP - compress.zlib:// 在 php-src 里可以找到和 compress.zlib:// 有关的代码 | code 注意到 STREAM_WILL_CAST&#xff0c;涉及到 cast 经常…

企业微信-通用开发参数回调设置

公司业务需要开发企业微信&#xff0c;注册三方服务商审核通过后&#xff0c; 开始配置开发信息。本篇中记录在调试url验证中遇到错误及解决方式。 目录 准备工作 下载php加解密库 下载文件说明 设置白名单 设置路径 参数说明 设置ip 回调处理 回调类型&#xff1a; …

【C++】布隆过滤器简单操纵模拟以及常见题目

&#x1f30f;博客主页&#xff1a; 主页 &#x1f516;系列专栏&#xff1a; C ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ &#x1f60d;期待与大家一起进步&#xff01; 文章目录 前言一、求下标仿函数的建议二、布隆过滤器代码面试题1.近似算法&#xff1a;2.精确算…

Leetcode 409. 最长回文串

文章目录 题目代码&#xff08;9.24 首刷自解&#xff09; 题目 Leetcode 409. 最长回文串 代码&#xff08;9.24 首刷自解&#xff09; class Solution { public:int longestPalindrome(string s) {unordered_map<char, int> mp;for(char c : s) mp[c];int res 0;int…

【算法思想-排序】排序数组-力扣 912 题

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kuan 的首页,持续学…

nexus 私服 拉不了 jar 包,报 Not authorized

如果你排查了所有情况,并且确定账号密码都没问题,路径也正确,并且setting.xml都配置正确了 可以看下是不是这个原因

5、SpringBoot_热部署

六、热部署 1.热部署概述 概述&#xff1a;程序更改后&#xff0c;不需要重新启动服务器也能够实现动态更新 springboot 项目如何实现热部署&#xff1f; tomcat 已经内置到项目容器中了希望tomcat监听外部程序变化通过新建一个程序来监控你代码的变化 2.依赖导入 依赖 <…

2023蓝帽杯半决赛misc题目复现

后续会逐渐完善&#xff1a; misc--排排坐吃果果 我真是无大语了&#xff0c;对于我的死脑筋&#xff0c;文件一打开是一片白色&#xff0c;但是点开单元格会看到里面有数字&#xff0c;我想到了修改单元格的格式&#xff0c;就是没想到转换字体的颜色&#xff0c;对此我表示…

构建基于neo4j知识图谱、elasticsearch全文检索的数字知识库

前言&#xff1a; 在数字化时代&#xff0c;知识库的建设正逐渐成为企业、学术机构和个人的重要资产。本文将介绍如何使用neo4j和elasticsearch这两种强大的数据库技术来构建知识库&#xff0c;并对其进行比较和探讨。 技术栈&#xff1a; springbootvueneo4jelasticsearch…

map的一些测试-string键的查找

主要区别在于声明map的时候多了一个less<> #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #include <map> #include <chrono> using namespace std; class spender { public:spender(string strfun) :strfun(strfun…