动态规划算法(2)最长回文子串详解

news2024/11/30 0:52:04

文章目录

  • 最长回文字串
    • 动态规划
    • 代码示例

前篇:
(1)初识动态规划

最长回文字串

传送门:
https://leetcode.cn/problems/longest-palindromic-substring/description/

给你一个字符串 s,找到 s 中最长的回文子串。
s ="babab”
结果:“babab”

解析,这是一道典型的动态规划的问题,但是如果你不知道动态规划,你会怎么做 ?

你可能会想到:依次截取字符串s中的每一个子字符串,然后每一次都比较这个子字符串是不是回文的,并且记录字串最大长度和起始下标,这样做固然可以。

但是!! 它的时间复杂度会非常高,把字符串切成每一种不同的字符串,仅仅 “babab” 这五个长度的字符串就有这些可能性,我们需要每次都判断它是否是回文字符串,并且记录,这对于leetcode的测试案例来说是不可能完成的!
他的时间复杂度是O(n^3),在check判断回文里还需要进行一次完整的遍历。因此这种方式是不要可行的。

b
ba
bab
baba
babab
a
ab
aba
abab
b
ba
bab
a
ab
b

完整代码如下:

class Solution {
public:
    bool check(string s)
    {
        int left=0,right=s.size()-1;
        while (left<right)
        {
            if (s[left++]!=s[right--])
            {
                return false;
            }
        }
        return true;
    }
    string longestPalindrome(string s) {
        int len=0;
        int max_len=0;
        int start=0;
        for (int i=0;i<s.size();i++)
        {
            len=1;
            for (int j=i;j<s.size();j++)
            {
                string temp=s.substr(i,len);
                if (check(temp))
                {
                    if (len>max_len)
                    {
                        max_len=len;
                        start=i;
                    }
                }
                len++;
            }

        }
        return s.substr(start,max_len);        
    }
};

动态规划

既然是回文串,那么我们不妨想一想回文串有什么性质呢?

以 “babab” 为例。

如果它是回文串,并且长度大于 2,那么将它首尾的两个字母去除之后,它仍然是个回文串: aba是回文串 —> babab 也是回文串,因为首位两个b也相等。

我们可以用 P(i,j)表示字符串s中 从 i 到 j 的字符组成的串,用s[i : j]表示这个串。


P(i,j)可以表示两种情况:

  1. true: s[i] -> s[j] 构成回文的字符串。
  2. false:非法的情况。

其中false非法的的情况又分为这两种情况:

  1. s[i] -> s[j] 本身不构成回文串
  2. i,j下标不合法,在动态规划中我们需要经常处理下标越界的情况。

状态转移方程:

P(i,j)=P(i+1,j-1)+ (s[i] == s[j])

也就是说:判断第s[i] -> s[j] 构成的字串是否是回文串

  1. 需要判断第 s[i+1] -> s[j-1] 是否构成回文串
  2. 判断第 s[i] 和 s[j] 首尾两个字符是否相同。
    在这里插入图片描述

我们来进一步分析这个状态转移的公式:

前提准备:

  • 我们创建dp二维数组表示他们的不同i,j时,i,j所对应的首尾子串的回文状态。
  • 我们首先从列开始,从上往下填充;然后从行开始,从左往右填充。
  1. 当 i==j的时候,可以看作左边界与有边界相等,因此这个字串就一个字符,这一定构成回文串,所以dp对角线元素一定是true,即是回文串
  2. (0,1)表示的字符串长度等于2,即比较s[0] == s[1],很明显不同,填充为 false
  3. (0,2)的状态可以由首尾两个字符的相等性(true)与他们缩小之后的子串(true)判断,bab显然是回文的,因此可以由状态方程之前存储的值得到(0,2)的状态,填充为true
  4. (1,2),他们的长度等于2,而且两个字符不相等,填充为false.
  5. (0,3):由(1,2)+ s[0]==s[3] 的状态确定。(1,2)已经被填充为了是false,因此只需比较首尾字符是否相等即可,填充为false
  6. (1,3):由(2,2)+s[1]==s[3]状态确定,填充为true
  7. (2,3):长度等于2,只需比较 s[2]==s[3],填充为false
  8. (0,4):由(1,3)+ s[0]==s[4]的状态确定。(1,3)已经被填充了true,再加上首尾字符相同,所以填充true
  9. (1,4):由(2,3)+ s[1]==s[4]状态确定,填充为false
  10. (2,4):由(3,3)+ s[2]==s[4]状态确定,填充为true
  11. (3,4):长度小于2,只需比较 s[3]==s[4],填充为false

在这里插入图片描述


代码示例

class Solution {
public:
    string longestPalindrome(string s) {
        int n=s.size();
        //---------------Tip1
        if (n==1)
        {
            return s;
        }
        vector<vector<bool>> dp(n,vector<bool>(n));
        for (int i=0;i<n;i++)
        {
            dp[i][i]=true;  //对角线一定为true,相等于i==j
        }
        int start=0,max_len=1;
        //---------------Tip2
        for (int L=2;L<=n;L++)
        {
            for (int i=0;i<n;i++)
            {
            	//--------------Tip3
                //j-i+1=L
                int j=L+i-1;
                if (j>=n)
                {
                //右边界越界,直接退出
                    break;
                }
                //首尾相等,填充true。那么不相等就填充false
                if (s[i]!=s[j])
                {
                    dp[i][j]=false;
                }
                else
                {	//到了这个else里面,那么就意味着 s[i] == s[j] 子串首尾字符是相同的
                	// j - i < 3: 填充长度小于等于3的字符串的状态
                	//长度小于等于3的长度为2的为true( s[i]=s[j])
                	//长度为3也是true(s[i] ==s[j]) && 中间的单独的字符,一定构成回文串。
                    if (j-i<3)
                    {
                        dp[i][j]=true;
                    }
                    else
                    {//长度较长的则进入状态转换,由之前的状态转换得到。
                        dp[i][j]=dp[i+1][j-1];
                    }
                }
                //每次都要记录目标子串的长度与坐标
                //1. 如果以 i为左,j为右的子串是dp[i][j]=true,则它是一个回文串
                //并且长度大于之前记录的子串,则更新此最长的回文串。
                if (dp[i][j] && j-i+1>max_len)
                {
                    max_len=j-i+1;
                    start=i;
                }
            }
        } 
        //根据得到的左边界坐标和长度得到目标子串
        return s.substr(start,max_len);
    }
};

需要注意的几个细节:

  • Tip1. 当我们的s只有一个字符时,它一定是回文串,直接返回即可。
  • Tip2. 我们根据子串的长度进而获得指定长度的子串进行依次填充dp数组,首先L 为2时,填充长度为2的子串的回文性;其次L为3,填充长度为3的字串…一直到最后L为5,填充长度为5的子串的回文性,本例中长度为5的字串即是 s 串本身。
  • Tip3. 我们都知道,想要得到一个子串就需要某个特定的左边界 i 以及子串的长度,这样我们就能确定一个子串。在本题中我们不仅要获得这两个元素还要获得 这个子串的右边界j:因为我们需要判断子串首尾两个字符是否相等,我们可以由 j - i + 1 = L(子串右边界下标 - 左边界下标 + 1 = 字串的长度),得到 j=L+i-1 ,即对应的右边界下标

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

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

相关文章

大数据学习:使用Java API操作HDFS

文章目录一、创建Maven项目二、添加依赖三、创建日志属性文件四、在HDFS上创建文件五、写入HDFS文件1、将数据直接写入HDFS文件2、将本地文件写入HDFS文件六、读取HDFS文件1、读取HDFS文件直接在控制台显示2、读取HDFS文件&#xff0c;保存为本地文件一、创建Maven项目 二、添加…

Spring Security 中重要对象汇总

前言 已经写了好几篇关于 Spring Security 的文章了&#xff0c;相信很多读者还是对 Spring Security 的云里雾里的。这是因为对 Spring Security 中的对象还不了解。本文就来介绍介绍一下常用对象。 认证流程 SecurityContextHolder 用户认证通过后&#xff0c;为了避免用…

【JavaWeb】Servlet系列 --- HttpServlet【底层源码分析】

HttpServlet一、什么是协议&#xff1f;什么是HTTP协议&#xff1f;二、HTTP的请求协议&#xff08;B -- > S&#xff09;1. HTTP的请求协议包括4部分&#xff08;记住&#xff09;2. HTTP请求协议的具体报文&#xff1a;GET请求3. HTTP请求协议的具体报文&#xff1a;POST请…

生成式模型和判别式模型

决策函数Yf(x)Y f(x)Yf(x)或者条件概率分布 P(Y∣X)P(Y|X)P(Y∣X) 监督学习的任务都是从数据中学习一个模型&#xff08;也叫做分类器&#xff09;&#xff0c;应用这一模型&#xff0c;对给定的输入xxx预测相应的输出YYY,这个模型的一般形式为:决策函数Yf(x)Y f(x)Yf(x)&…

java 每日一练(6)

java 每日一练(6) 文章目录单选不定项选择题编程题单选 1.关于抽象类与最终类&#xff0c;下列说法错误的是&#xff1f;   A 抽象类能被继承&#xff0c;最终类只能被实例化。   B 抽象类和最终类都可以被声明使用   C 抽象类中可以没有抽象方法&#xff0c;最终类中可以没…

Bean 管理(工厂bean)

IOC操作Bean 管理&#xff08;FactoryBean&#xff09; 下面是在Bean 管理&#xff08;工厂bean&#xff09;之前的学习&#xff0c;基于xml方式注入集合并实现 基于xml方式注入集合并实现 &#xff1a;http://t.csdn.cn/H0ipR Spring 有两种类型bean&#xff0c;一种普通bean…

第五章. 可视化数据分析分析图表—图表的常用设置2

第五章. 可视化数据分析分析图 5.2 图表的常用设置2 本节主要介绍图表的常用设置&#xff0c;设置标题和图例&#xff0c;添加注释文本&#xff0c;调整图表与画布边缘间距以及其他设置。 7.设置标题&#xff08;title&#xff09; 1).语法&#xff1a; matplotlib.pyplot.ti…

iOS15适配 UINavigationBar和UITabBar设置无效,变成黑色

今天更新了xcode13&#xff0c;运行项目发现iOS15以上的手机导航栏和状态栏之前设置的颜色等属性都不起作用了&#xff0c;都变成了黑色&#xff0c;滚动的时候才能变成正常的颜色&#xff0c;经确认得用UINavigationBarAppearance和UITabBarAppearance这两个属性对导航栏和状态…

开发SpringBoot+Jwt+Vue的前后端分离后台管理系统VueAdmin - 前端笔记

一个spring security jwt vue的前后端分离项目&#xff01;综合运用&#xff01; 关注公众号 MarkerHub&#xff0c;回复【 VueAdmin 】可以加群讨论学习、另外还会不定时安排B站视频直播答疑&#xff01; 首发公众号&#xff1a;MarkerHub 作者&#xff1a;吕一明 视频讲解&…

半年卖8万吨辣条,卫龙再闯IPO

“辣条大王”卫龙美味全球控股有限公司&#xff08;下称“卫龙”&#xff09;于11月23日重新递表&#xff0c;继续冲刺“辣条第一股”。 作为千禧一代撑起的童年“神话”&#xff0c;卫龙的上市之路却波折重重&#xff1b;它曾于2021年5月、2021年11月及此次重启IPO。 卫龙是…

线程池ThreadPoolExecutor

线程池的生命周期 private final AtomicInteger ctl new AtomicInteger(ctlOf(RUNNING, 0)); ThreadPoolExecutor使用一个ctl变量代表两个信息&#xff0c;线程池的运行状态 (runState) 和 线程池内有效线程的数量 (workerCount)&#xff0c;高三位表示状态。 workerCount&am…

MySQL 数据库存储引擎

目录 一、存储引擎简介 二、MyISAM存储引擎 1、MylSAM介绍 2、MyISAM表支持3种不同的存储格式 3、MylSAM的特点 4、MyISAM使用的生产场景 三、InnoDB存储引擎 1、InnoDB介绍 2、InnoDB的特点 3、InnoDB适用生产场景 4、MyISAM和InnoDB的区别 四、查看和修改存储引擎…

CloudAlibaba - Nacos服务注册与配置中心

文章目录一.CloudAlibaba简介1. 介绍2. 依赖3. 主要组件4. 资料文档二.Nacos服务注册与发现1. 简介2. Nacos安装3. Nacos服务注册3.1 注册服务生产者3.2 服务消费者注册和负载4. Nacos服务中心对比三.Nacos配置中心1. 基础配置搭建2. Nacos中添加配置信息2.1 Nacos中的匹配规则…

单商户商城系统功能拆解40—分销应用—分销设置

单商户商城系统&#xff0c;也称为B2C自营电商模式单店商城系统。可以快速帮助个人、机构和企业搭建自己的私域交易线上商城。 单商户商城系统完美契合私域流量变现闭环交易使用。通常拥有丰富的营销玩法&#xff0c;例如拼团&#xff0c;秒杀&#xff0c;砍价&#xff0c;包邮…

前端js传入Long类型精度丢失解决办法

目录 问题背景 解决过程 1.对比数据库数据 2.查询资料 解决方法 问题背景 在进行业务开发的 时候发现更新数据库中的一条数据没有成功&#xff0c;查看SQL日志发现SQL正常执行无错误信息&#xff0c;但是受影响行数为0&#xff0c;但是数据是从前端传过来的 &#xff0c;一…

面试官:MySQL explain你会关注哪些字段

编号字段名称字段说明值重要性1idSELECT查询序号&#xff0c;即为sql语句执行的顺序。数字越大越优先执行1、2、3...★★2select_typeselect类型&#xff0c;用于区分是普通查询、子查询或联合查询simple&#xff1a;简单的查询&#xff0c;不含子查询和联合查询primary&#x…

玩机搞机---卸载内置软件 无root权限卸载不需要的软件 安全卸载

目前的机型出厂都会自带一些内置软件。个别的还无法删除。各种弹窗广告 等等占用系统资源。操作卡顿。而很多卸载软件有需要root权限才可以运行。那么有没有方法可以无需root就可以卸载内置软件呢。答案是有的。而且方法很多。今天的帖子就列举一下常用的操作步骤 操作前了解的…

自己搭建网站【搭建网站】

现在搭建网站也不只是企业公司商家的专利&#xff0c;很多个人用户都想拥有属于自己的一个网站。那么怎么自己搭建网站呢&#xff1f;下面给大家简单说说。 一、申请域名 域名是访问网站的地址&#xff0c;这是必须要有的&#xff0c;可以在域名服务商网站上申请。申请域名需要…

科普-鸟哥私房菜中的“od -t oCc /etc/issue”的意思

文章摘录展示 看到这里是困惑的&#xff0c;毕竟看这里没有写oCc 的格式 既然文章不能找到答案&#xff0c;那么&#xff0c;我们看详细的说明吧&#xff08;man od&#xff09; 如图&#xff0c;想必我们应该很清楚了 &#xff0c;o[size] 对应的参数oC &#xff0c;就可以很…

2022.11.27Find a way HDU - 2612

Pass a year learning in Hangzhou, yifenfei arrival hometown Ningbo at finally. Leave Ningbo one year, yifenfei have many people to meet. Especially a good friend Merceki. Yifenfei’s home is at the countryside, but Merceki’s home is in the center of city.…