【动态规划算法】-回文串问题题型(34-40题)
💖作者:小树苗渴望变成参天大树🎈
🎉作者宣言:认真写好每一篇博客💤
🎊作者gitee:gitee✨
💞作者专栏:C语言,数据结构初阶,Linux,C++ 动态规划算法🎄
如 果 你 喜 欢 作 者 的 文 章 ,就 给 作 者 点 点 关 注 吧!
文章目录
- 前言
- 第三十五题:[647. 回文子串](https://leetcode.cn/problems/palindromic-substrings/)
- 第三十六题:[5. 最长回文子串](https://leetcode.cn/problems/longest-palindromic-substring/)
- 第三十七题:[1745. 分割回文串 IV](https://leetcode.cn/problems/palindrome-partitioning-iv/)
- 第三十八题:[132. 分割回文串 II](https://leetcode.cn/problems/palindrome-partitioning-ii/)
- 第三十九题:[516. 最长回文子序列](https://leetcode.cn/problems/longest-palindromic-subsequence/)
- 第四十题:[1312. 让字符串成为回文串的最少插入次数](https://leetcode.cn/problems/minimum-insertion-steps-to-make-a-string-palindrome/)
前言
今天博主来讲解动态规划的另一个题型就是回文串问题,这个系列的问题,套路差不多,首先大家要了解什么是回文串,就是在原串中选出连续的一个子串,判断是不是回文的即可,接下来我们将会一题题的给大家进行讲解,话不多说,我们开始进入正文
第三十五题:647. 回文子串
题目解析:
动态规划算法:
1. 状态表示:经验+题目要求
做这个系列的问题我们就是将子串是否为回文的结果放到dp表里面,因为你要插进来的字符是否和前面构成回文,首先保证前面的子串是回文才行,所以我们的dp表,需要一个二维的,表示回文子串的起始和结尾
dp[i][j]表示:以i,j位置为结尾的子串是否为回文子串,(i<=j)
2. 状态转移方程:
3. 初始化:保证数组不越界
我们j是大于等于i的,但是会越界的情况已经单独拿出来分析了,所以不用初始化
4. 填表顺序:
所以我们要从上往下进行填表
5. 返回值:
返回为真的个数就可以了
代码实现:
class Solution {
public:int countSubstrings(string s) {int n=s.size();vector<vector<bool>> dp(n,vector<bool>(n));int count=0;for(int i=n-1;i>=0;i--){for(int j=i;j<n;j++){if(s[i]==s[j]){if(i==j||i+1==j)dp[i][j]=true;elsedp[i][j]=dp[i+1][j-1];}if(dp[i][j]==true) count++;}}return count;}
};
运行结果:
这一题可以说是为后面的题目做铺垫,相当于一个引子
第三十六题:5. 最长回文子串
题目解析:
此题就是找到最长回文子串,做法和上一题一模一样,先用dp表存放是否构成回文串,有起始位置和结束位置就可以算出来长度。
动态规划算法:
1. 状态表示:经验+题目要求
dp[i][j]表示:以i,j位置为结尾的子串是否为回文子串,(i<=j)
2. 状态转移方程:
每次填完dp表的时候,就计算一个长度,i表示起始位置,j表示结束位置,长度为j-i+1,
3. 初始化:保证数组不越界
我们j是大于等于i的,但是会越界的情况已经单独拿出来分析了,所以不用初始化
4. 填表顺序:
所以我们要从上往下进行填表
5. 返回值:
将长度和起始位置算出来,通过substr来获取子串返回即可
代码实现:
class Solution {
public:string longestPalindrome(string s) {int n=s.size();vector<vector<bool>> dp(n,vector<bool>(n));int len=1,start=0;for(int i=n-1;i>=0;i--){for(int j=i;j<n;j++){if(s[i]==s[j]){if(i==j||i+1==j)dp[i][j]=true;elsedp[i][j]=dp[i+1][j-1];}if(dp[i][j]&&len<j-i+1)//选出最长的回文子串{len=j-i+1,start=i;}}}return s.substr(start,len);}
};
运行结果:
第三十七题:1745. 分割回文串 IV
题目解析:
这个题目还是非常的好理解的,因为有固定的数,分成三个回文子串。
先将子串是否为回文子串放到dp表里面,然后再枚举dp表就行了。
这题我就不详细讲解了,直接上代码了
代码实现:
class Solution {
public:bool checkPartitioning(string s) {int n=s.size();vector<vector<bool>> dp(n,vector<bool>(n));int count=0;for(int i=n-1;i>=0;i--)//将子串是否是文回放到dp表里面{for(int j=i;j<n;j++){if(s[i]==s[j]){if(i==j||i+1==j)dp[i][j]=true;elsedp[i][j]=dp[i+1][j-1];}}}for(int i=1;i<n-1;i++)//第一个至少要留出来一个位置{for(int j=i;j<n-1;j++)//最后一个也要留出来一个位置{if(dp[i][j]&&dp[0][i-1]&&dp[j+1][n-1])return true;}}return false;}
};
运行结果:
第三十八题:132. 分割回文串 II
题目解析:
动态规划算法:
1. 状态表示:经验+题目要求
我们以某一个位置来考虑问题:
dp[i]表示:从0开始到以i位置为结尾的子串中将子串分割成每个部分都是回文子串的最少的切割次数
2. 状态转移方程:
这题和单词拆分的思想有点像,再[0,i]区间中选择一个j,此时j位置就是最后一刀切割的地方,前面切割的次数,加上最后一刀就是总次数
3. 初始化:保证数组不越界
j-1是不会越界的,j>0的。
因为要保证第一次求最小值的时候不能干扰到选择dp[j-1]+1,所以不初始化,就为01,那么最小值一直都是0,所以干脆就初始化为最大值。
4. 填表顺序:
从左往右
5. 返回值:
返回dp[n-1];
代码实现:
class Solution {
public:int minCut(string s) {int n=s.size();vector<vector<bool>> isPal(n,vector<bool>(n));int count=0;for(int i=n-1;i>=0;i--)//将子串是否是文回放到dp表里面{for(int j=i;j<n;j++){if(s[i]==s[j]){if(i==j||i+1==j)isPal[i][j]=true;elseisPal[i][j]=isPal[i+1][j-1];}}}//创建dp表+初始化vector<int> dp(n,INT_MAX);for(int i=0;i<n;i++){if(isPal[0][i]){dp[i]=0;}else{for(int j=1;j<=i;j++)if(isPal[j][i])dp[i]=min(dp[j-1]+1,dp[i]);} } //返回值return dp[n-1];}
};
运行结果:
第三十九题:516. 最长回文子序列
题目解析:
动态规划算法:
1. 状态表示:经验+题目要求
dp[i]表示:以i位置元素为结尾的子序列中最长的回文子序列的长度
上面的状态表示不行,我们要重新定义状态表示,定义两个位置
dp[i][j]表示:s字符串里面【i,j】子区间内的所有子序列中最长回文子序列的长度
2. 状态转移方程:
3. 初始化:保证数组不越界
我们看到会使用到使用到j-1或者i+1位置的值
4. 填表顺序:
我们会使用到
dp[i+1][j]正下方的值
dp[i][j-1]左边的值
dp[i+1][j-1]左下方的值
从下往上,从左往右
5. 返回值:
返回整个字符串区间dp[0][n-1]
代码实现:
class Solution {
public:int longestPalindromeSubseq(string s) {int n=s.size();//创建dp表vector<vector<int>> dp(n,vector<int>(n));for(int i=n-1;i>=0;i--)//从下往上{dp[i][i]=1;//此情况肯定是相等的for(int j=i+1;j<n;j++)//从左往右{if(s[i]==s[j])dp[i][j]=dp[i+1][j-1]+2; //放在一起考虑了 else dp[i][j]=max(dp[i][j-1],dp[i+1][j]);//不相等的时候}}//返回值return dp[0][n-1];}
};
运行结果:
第四十题:1312. 让字符串成为回文串的最少插入次数
题目解析:
这题还是按照上一题的分析方式一样,选择一个区间去分析。
动态规划算法:
1. 状态表示:经验+题目要求
dp[i][j]表示:s字符串以[i,j]区间内想要变成回文串要插入的最少次数
2. 状态转移方程:
3. 初始化:保证数组不越界
根据上题的分析无需初始化
4. 填表顺序:
从下往上,从左往右
5. 返回值:
dp[0][n-1];
代码实现:
class Solution {
public:int minInsertions(string s) {int n=s.size();//创建dp表vector<vector<int>> dp(n,vector<int>(n));for(int i=n-1;i>=0;i--)//从下往上for(int j=i+1;j<n;j++)//从左往右,从i+1位置,因为i==j的情况为0不需要考虑if(s[i]==s[j])dp[i][j]=dp[i+1][j-1];else dp[i][j]=min(dp[i][j-1],dp[i+1][j])+1;//返回值return dp[0][n-1];}
};
运行结果:
到这里我们的回文串问题就讲解完毕了,这类题型之间的练习都挺大了,上一题的经验可以大量的运用到下一题上,这给我们的学习也带来了很大的帮助,但凡不是一题题做过来的,那么后面的每一题都很难想到,所以我们要善于总结,这样下次单独碰到一题就不会没有头绪,好了,我们今天的题目就先讲解到这里了,我们下篇介绍关于两个数组的dp问题。
相关文章:

【动态规划算法】-回文串问题题型(34-40题)
💖作者:小树苗渴望变成参天大树🎈 🎉作者宣言:认真写好每一篇博客💤 🎊作者gitee:gitee✨ 💞作者专栏:C语言,数据结构初阶,Linux,C 动态规划算法🎄 如 果 你 …...

STM32基础回顾
文章目录 单片机编程的原理GPIO中断EXTI外部中断定时器中断、串口中断 定时器定时器中断配置过程通用定时器输出比较功能:PWM波的生成定时器的输入捕获功能主从触发模式PWMI模式 定时器的编码器接口 DMA简介通信接口USART软件配置流程:1、仅发数据的配置…...

如何解决电脑无声问题:排除故障的几种常见方法
大家好,今天我们来讨论一下处理电脑没有声音的故障。当你突然发现电脑静音无声时,需要逐步排除可能的问题,但总体而言,声音故障是相对容易解决的。接下来,我们将介绍一些排除电脑无声问题的方法。 第一步:…...

Apache RocketMQ 命令注入
漏洞简介 RocketMQ 5.1.0及以下版本,在一定条件下,存在远程命令执行风险。RocketMQ的NameServer、Broker、Controller等多个组件外网泄露,缺乏权限验证,攻击者可以利用该漏洞利用更新配置功能以RocketMQ运行的系统用户身份执行命令…...

二、搜索与图论6:Dijkstra 模板题+算法模板(Dijkstra求最短路 I, Dijkstra求最短路 II,1003 Emergency)
文章目录 算法模板Dijkstra题目代码模板朴素dijkstra算法堆优化版dijkstra 树与图的存储(1) 邻接矩阵:(2) 邻接表:关于e[],ne[],h[]的理解 关于堆的原理与操作 模板题Dijkstra求最短路 I原题链接题目思路题解 Dijkstra求最短路 II原题链接题目思路题解 1…...

ROS2学习(四)进程,线程与节点的关系
节点与节点执行器 节点,英文是node,在ROS2中,节点是一个抽象的实体,它可以代表某种或某类特定功能的抽象集合体,它可以存在于进程中,也可以存在于线程中。所有ROS2的基础功能最基础的载体是节点,所有的通信…...

【物联网】DMA传输原理与实现详解(超详细)
DMA(Direct Memory Access,直接内存访问)是一种计算机数据传输方式,允许外围设备直接访问系统内存,而无需CPU的干预。 文章目录 Part 1: DMA的工作原理配置阶段:数据传输阶段: Part 2: DMA数据…...

Java类集框架(二)
目录 1.Map(常用子类 HashMap,LinkedHashMap,HashTable,TreeMap) 2.Map的输出(Map.Entry,iterator,foreach) 3.数据结构 - 栈(Stack) 4.数据结构 - 队列(Q…...

爬虫008_流程控制语句_if_if else_elif_for---python工作笔记026
然后我们再来看一下这里的,判断,可以看到 再看一个判断,这里的布尔类型 第二行有4个空格,python的格式 注意这里,输入的age是字符串,需要转一下才行 int可以写到int(intput("阿斯顿法师打发地方")) 这样也可以...

【随笔】五周年创作纪念日
今天收到了 CSDN 的创作五周年提示,正好前几天(7.31)我也成功申请了 CSDN 博客专家,趁这个机会分享一下这几年写博客的感受吧 机缘 关注我比较久的读者应该知道我是从学传统工科半路出家搞计算机的,这里的经历还是比…...

7_分类算法—逻辑回归
文章目录 逻辑回归:1 Logistic回归(二分类问题)1.1 sigmoid函数1.2 Logistic回归及似然函数(求解)1.3 θ参数求解1.4 Logistic回归损失函数1.5 LogisticRegression总结 2 Softmax回归(多分类问题࿰…...

【计算机网络】应用层协议 -- DNS协议
文章目录 1. DNS背景2. 域名简介3. 域名解析过程4. 使用dig查看DNS过程 1. DNS背景 DNS(Domain Name System,域名系统)协议,是一个用来将域名转化为IP地址的应用层协议。 TCP/IP当中通过IP地址和端口号的方式,来确定…...
ES6 - 数组新增的一些常用方法
文章目录 1,Array.from()2,Array.of()3,find(),findIndex(),findLast()和findLastIndex()4,Array.fill()5,keys(),values() 和 entries()6,Array.includes()7,…...

【BEV感知】3-BEV开源数据集
3-BEV开源数据集 1 KITTI1.1 KITTI数据怎么采集?1.2 KITTI数据规模有多大?1.3 KITTI标注了哪些目标?1.4 转换矩阵1.5 标签文件 2 nuScenes2.1 nuScenes Vs KITTI2.2 标注文件 1 KITTI KITTI 1.1 KITTI数据怎么采集? 通过车载相机、激光雷达等传感器采集。 只提供了相机正…...

Kafka-Broker工作流程
kafka集群在启动时,会将每个broker节点注册到zookeeper中,每个broker节点都有一个controller,哪个controller先在zookeeper中注册,哪个controller就负责监听brokers节点变化,当有分区的leader挂掉时,contro…...
第八篇-Tesla P40+ChatGLM2+LoRA
部署环境 系统:CentOS-7CPU: 14C28T显卡:Tesla P40 24G驱动: 515CUDA: 11.7cuDNN: 8.9.2.26目的 验证P40部署可行性,只做验证学习lora方式微调创建环境 conda create --name glm-tuning python3.10 conda activate glm-tuning克隆项目 git clone http…...

调用feign返回错误的数据
bug描述: 在一个请求方法中会调用到feign去获取其他的数据。 List<Demo> list aaaFeignApi.getData(personSelectGetParam);在调用的时候,打断点到feign的地方,数据是存在的,并且有15条。但是返回到上面代码的时候数据就…...

【Spring】(二)从零开始的 Spring 项目搭建与使用
文章目录 前言一、Spring 项目的创建1.1 创建 Maven 项目1.2 添加 Spring 框架支持1.3 添加启动类 二、储存 Bean 对象2.1 创建 Bean2.1 将 Bean 注册到 Spring 容器 三、获取并使用 Bean 对象3.1 获取Spring 上下文3.2 ApplicationContext 和 BeanFactory 的区别3.3 获取指定的…...

redis五种数据类型介绍
、string(字符串) 它师最基本的类型,可以理解为Memcached一模一样的类型,一个key对应一个value。 注意:一个键最大能存储 512MB。 特性:可以包含任何数据,比如jpg图片或者序列化的对象,一个键最大能存储512…...

【JavaEE】Spring Boot - 项目的创建和使用
【JavaEE】Spring Boot 开发要点总结(1) 文章目录 【JavaEE】Spring Boot 开发要点总结(1)1. Spring Boot 的优点2. Spring Boot 项目创建2.1 下载安装插件2.2 创建项目过程2.3 加载项目2.4 启动项目2.5 删除一些没用的文件 3. Sp…...

第19节 Node.js Express 框架
Express 是一个为Node.js设计的web开发框架,它基于nodejs平台。 Express 简介 Express是一个简洁而灵活的node.js Web应用框架, 提供了一系列强大特性帮助你创建各种Web应用,和丰富的HTTP工具。 使用Express可以快速地搭建一个完整功能的网站。 Expre…...
React Native 导航系统实战(React Navigation)
导航系统实战(React Navigation) React Navigation 是 React Native 应用中最常用的导航库之一,它提供了多种导航模式,如堆栈导航(Stack Navigator)、标签导航(Tab Navigator)和抽屉…...
QMC5883L的驱动
简介 本篇文章的代码已经上传到了github上面,开源代码 作为一个电子罗盘模块,我们可以通过I2C从中获取偏航角yaw,相对于六轴陀螺仪的yaw,qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...

Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
镜像里切换为普通用户
如果你登录远程虚拟机默认就是 root 用户,但你不希望用 root 权限运行 ns-3(这是对的,ns3 工具会拒绝 root),你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案:创建非 roo…...

C# 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...

push [特殊字符] present
push 🆚 present 前言present和dismiss特点代码演示 push和pop特点代码演示 前言 在 iOS 开发中,push 和 present 是两种不同的视图控制器切换方式,它们有着显著的区别。 present和dismiss 特点 在当前控制器上方新建视图层级需要手动调用…...

华为OD机考-机房布局
import java.util.*;public class DemoTest5 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseSystem.out.println(solve(in.nextLine()));}}priv…...
比较数据迁移后MySQL数据库和OceanBase数据仓库中的表
设计一个MySQL数据库和OceanBase数据仓库的表数据比较的详细程序流程,两张表是相同的结构,都有整型主键id字段,需要每次从数据库分批取得2000条数据,用于比较,比较操作的同时可以再取2000条数据,等上一次比较完成之后,开始比较,直到比较完所有的数据。比较操作需要比较…...
6️⃣Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙
Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙 一、前言:离区块链还有多远? 区块链听起来可能遥不可及,似乎是只有密码学专家和资深工程师才能涉足的领域。但事实上,构建一个区块链的核心并不复杂,尤其当你已经掌握了一门系统编程语言,比如 Go。 要真正理解区…...