当前位置: 首页 > news >正文

动态规划解决leetcode上的两道回文问题(针对思路)

本期主讲的是使用动态规划去解决两道回文问题,分别是

647. 回文子串 - 力扣(LeetCode)

516. 最长回文子序列 - 力扣(LeetCode)

而不是leetcode5.最长回文子串,虽然这道题也是回文问题,也可以使用dp来解决,但是这道题本人认为与647回文子串思路相同,但这里也给出大致的解题思路。

这道题求的是连续子数组问题,而且是最长的回文的子数组,虽然它只有一个字符串,但是我们也使用二维dp数组,一维代表起始子数组,另一维是标识该子数组的终点,然后从后往前遍历字符串,两层循环,分别指子串起始和末尾,然后判断当前两个下标对应字符是否相等,然后再看中间的子数组是否回文,若是则更新答案。

这道题的思路其实就是我们要讲的回文子串的思路,只不过处理数据有些不同而已,求最长子数组就是记录当前多长,和之前的比较。而这道记录的是有多少回文子串,实际思路是相同的


第一道题 

先看代码:

class Solution {
public:int countSubstrings(string s) {int n=s.size();vector<vector<bool>>dp(n,vector<bool>(n,false));int res=0;for(int i=n-1;i>=0;--i){for(int j=i;j<n;++j){if(s[i]==s[j]){if(j-i<=1){dp[i][j]=true;++res;}else if(dp[i+1][j-1]){dp[i][j]=true;++res;}}}}return res;}
};

 我们以一个布尔类型二维数组做判断,循环就是刚才说的,从后往前,为什么从后往前,这是因为递推式的缘故,我们要判断当前字符相等的情况下,看一看该子串里面的内容是否回文,怎么看?它是比较i+1和j-1,所以一定要保证这一段已经被判断回文过了,所以我们才会选择从后往前遍历的方法,而内层循环你看着可能有些奇怪,但事实上它是正确的。因为如果两个下标对应相等还存在两种情况,一种是两下标指向同一字符,此时肯定回文,另一种是指向不同字符,但是这两个下标只差1,也就是说【a】和【aa】这两种情况的对应。我们都考虑了进来,当两下标相差大的时候,里面就会存有子数组我们才会进行判断,而此时判断的话,已经是之前遍历过的位置了,已经知道了它回不回文。

可能有读者会问,是否可以从字符串中间去进行向两边扩散的访问,以得到是否回文的情况,这一招很好用,对于判断整个字符串是否回文,以及判断整个字符串的最长回文子数组的长度,但是并不适用于本题,因为本题求解的是根据每一个下标的情况给出可能存在的所有回文的子数组的个数

而上面的那招中心探测,并不完全,如果使用中心探测至少要对每个子串都进行回文判断,所以无论如何你都逃脱不了,要判断每个子串的命运,而不是只在字符串的中心位置判断回文,而只On就能解决问题,你最少需要ON^2才可以解决问题。(至少我知道的方法里没有ON)

当然如果你是算法老鸟,你可以对二维数组进行一维的压缩,以解决减少时间复杂度的开销,但是通常我只会在简单的题解里才会想这种优化,稍稍复杂一点的题解,很容易出现各种问题,所以我一般倾向于解题,而不是给自己找麻烦。


第二道题

 最长回文子序列,是不是看起来和那道最长回文子串差不多,这道题只不过是求不连续的子序列最长的长度。

但我们这里的思路并不沿用上面的思路。尽量不要使用回文子串那道题目的做法,会使得代码难写且含义不清晰,因为那道做法dp含义是针对于连续子数组的,我们很难只用true或false来判断中间字符串的回文长度到底为多少,而且在本题中,这样的bool数组并没有太多价值,因为如果要存子序列的最长数多半情况是用到其他变量辅助存储的,而bool数组起到的作用只是判断这段子数组是否回文,与这道题实际的问题有点相违背。

这道题我们采用如下的思路:

class Solution {
public:int longestPalindromeSubseq(string s) {int n=s.size();vector<vector<int>>dp(n,vector<int>(n,0));for(int i=0;i<n;++i)dp[i][i]=1;for(int i=n-1;i>=0;--i){for(int j=i+1;j<s.size();++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];}
};

 dp数组的含义代表当前ij范围内最长的回文子序列多长,子序列可以不连续
当此时ij对应下标字符相等,则用两下标中间的子串范围的最大回文子序列长度+2,也就是把s【i】和s【j】都加入进来
如果此时不相等,则判断如果只加进一个能否取得最长回文子序列,也就是说dp【i】【j-1】和dp【i+1】【j】它们两个比较然后取最大值。
为什么这样呢?因为s【i】和s【j】中间的子数组应该是以s【i+1】和s【j-1】为下标的,所以左边加进来s【i】就是dp【i】【j-1】右边加进来j就是dp【i+1】【j】
别忘了根据递推式的含义,最后返回的应该是dp【0】【s.size()-1】
代表从0开始到结束最长的回文子序列

是不是有疑问了?为什么这道题的内部循环变得更奇怪了?要使用j=i+1的初始化,这样不会越界吗?

接下来我们来解释一下这样做的原因。

里层循环i+1,并不是无用的,不能设置为j=i
有两个原因一个是浅显易见的,是因为两个下标指向同一字符的情况我们已经初始完毕了,另一原因是如果从j=i进来那第一次填数会导致数组非法下标访问,而第二次进来虽然推导时候也有些奇怪,因为i和j拉开距离过小的缘故,i会走到j的位置,也就是i上一次遍历的位置,j会走到i本次起始遍历位置,虽然奇怪,但是也是遍历过的,所以答案也是对的。

没错前面的循环是为了初始化,因为至少它的最长回文子序列一定是1,我们要把它初始化一下,这也方便接下来dp数组的正确填数。而由于递推式的缘故,我们也不能使内层循环写成j=i开始,要不然一开始就会越界,这道题不像上一道题,上一道题是极可能存在相邻两个相等的情况,而且那道题的dp数组含义也与本题不同。那道题j=i的初始化由于前面的if判断,j<=i的时候会有一个缓冲,不会过早进到中间子数组的判断,这道题递推式不行,它不具备那样的缓冲。

那我们可以加一个那样的判断吗?判断两字符相等且j<=I然后填数做缓冲不就可以了吗?

好问题,可以

但是求子序列的含义是我们可以允许不相邻的子序列也构成答案,也就是说子序列答案是包含了两个字符相邻回文,也就是子序列包含子数组的所有情况,没必要写一个特式专门用来判断这一步骤,这显得有点繁琐了,子序列的代码应该更适用于大多数的情况。

但如果加一步那样的判别可以使一开始的判断回文看起来正常一些,j也不用从i+1开始判断了,也是一种好的题解思路。

那这道题可以从字符串中间进行下标的开始匹配吗?我认为有点问题,类中心探测法无论是一次匹配还是每个字符的分别匹配,本质上都是用来解决连续的子数组的问题,用在子序列可能有点麻烦,我没有试过,感兴趣的读者可以自己试一下。我认为应该有点难度。


都看到这里了如果对您有用的话别忘了一键三连哦,如果是互粉回访我也会做的!

大家有什么想看的题解,或者想看的算法专栏、数据结构专栏,可以去看看往期的文章,有想看的新题目或者专栏也可以评论区写出来,讨论一番,本账号将持续更新。
期待您的关注

相关文章:

动态规划解决leetcode上的两道回文问题(针对思路)

本期主讲的是使用动态规划去解决两道回文问题&#xff0c;分别是 647. 回文子串 - 力扣&#xff08;LeetCode&#xff09; 516. 最长回文子序列 - 力扣&#xff08;LeetCode&#xff09; 而不是leetcode5.最长回文子串&#xff0c;虽然这道题也是回文问题&#xff0c;也可以…...

使用人工智能自动测试 Flutter 应用程序

移动应用程序开发的增长速度比以往任何时候都快。几乎每个企业都需要移动应用程序来保持市场竞争力。由于像 React Native 这样的跨平台移动应用程序开发框架允许公司使用单一源代码和单一编程语言构建 iOS 和 Android 应用程序&#xff0c; Flutter是 Google 支持的另一个热门…...

四、程序员指南:数据平面开发套件

REORDER LIBRARY 重排序库提供了根据其序列号对mbuf进行重排序的机制。 16.1 操作 重排序库本质上是一个对mbuf进行重新排序的缓冲区。用户将乱序的mbuf插入重排序缓冲区&#xff0c;并从中提取顺序正确的mbuf。 在任何给定时刻&#xff0c;重排序缓冲区包含其序列号位于序列…...

Go 之 captcha 生成图像验证码

目前 chptcha 好像只可以生成纯数字的图像验证码&#xff0c;不过对于普通简单应用来说也足够了。captcha默认将store封装到内部&#xff0c;未提供对外操作的接口&#xff0c;因此使用自己显式生成的store&#xff0c;可以通过store自定义要生成的验证码。 package mainimpor…...

【Java从入门到大牛】多线程

&#x1f525; 本文由 程序喵正在路上 原创&#xff0c;CSDN首发&#xff01; &#x1f496; 系列专栏&#xff1a;Java从入门到大牛 &#x1f320; 首发时间&#xff1a;2023年11月18日 &#x1f98b; 欢迎关注&#x1f5b1;点赞&#x1f44d;收藏&#x1f31f;留言&#x1f4…...

UE5 C++报错:is not currently enabled for Live Coding

解决办法&#xff1a; 再次打开项目&#xff0c;以此法打开&#xff1a;...

mysql服务器数据同步

在Linux和Windows之间实现MySQL服务器数据的同步。下面是一些常见的方法和工具&#xff1a; 复制&#xff08;Replication&#xff09;&#xff1a;MySQL复制是一种常见的数据同步技术&#xff0c;可用于将一个MySQL服务器的数据复制到其他服务器。您可以设置主服务器&#xff…...

Docker Golang 开发环境搭建指南

Docker Golang 开发环境搭建指南 概述 在 Golang 开发中&#xff0c;搭建合适的开发环境是非常重要的。然而&#xff0c;由于 Golang 的跨平台特性&#xff0c;不同操作系统之间的配置差异可能会导致环境搭建过程变得复杂。为了简化这个过程并保持开发环境的一致性&#xff0…...

MFC保存窗口客户区为图片

首先的窗口输出一些内容&#xff1b; 菜单单击函数代码&#xff1b; void CgetmypicView::OnTestGetmypic() {// TODO: 在此添加命令处理程序代码HWND hwnd this->GetSafeHwnd();HDC hDC ::GetWindowDC(hwnd);//获取DC RECT rect;::GetClientRect(hwnd, &rect)…...

JAVA安全之Shrio550-721漏洞原理及复现

前言 关于shrio漏洞&#xff0c;网上有很多博文讲解&#xff0c;这些博文对漏洞的解释似乎有一套约定俗成的说辞&#xff0c;让人云里来云里去&#xff0c;都没有对漏洞产生的原因深入地去探究..... 本文从现象到本质&#xff0c;旨在解释清楚Shrio漏洞是怎么回事&#xff01…...

有Mac或无Mac电脑通用的获取安卓公钥的方案

从2023年9月开始&#xff0c;所有上架应用市场的app都需要进行APP备案。 其中后端服务器在阿里云的可以在阿里云备案&#xff0c;后端服务器在腾讯云的可以在腾讯云备案。但无论你是在什么云厂商里做备案&#xff0c;无一例外的是&#xff0c;无论是上架安卓应用还是上架IOS应…...

电池故障估计:Realistic fault detection of li-ion battery via dynamical deep learning

昇科能源、清华大学欧阳明高院士团队等的最新研究成果《动态深度学习实现锂离子电池异常检测》&#xff0c;用已经处理的整车充电段数据&#xff0c;分析车辆当前或近期是否存在故障。 思想步骤&#xff1a; 用正常电池的充电片段数据构造训练集&#xff0c;用如下的方式构造…...

微服务和Spring Cloud Alibaba介绍

1、微服务介绍 1.1 系统架构演变 随着互联网的发展&#xff0c;网站应用的规模也在不断的扩大&#xff0c;进而导致系统架构也在不断的进行变化。从互联网早起到现在&#xff0c;系统架构大体经历了下面几个过程: 单体应用架构 —> 垂直应用架构 —> 分布 式架构—>…...

【js】 lodash命名转换和封装

▒ 目录 ▒ &#x1f6eb; 导读需求开发环境 1️⃣ lodash转换函数h3与underscore比较 2️⃣ 实战&#xff1a;对象属性名转换函数封装单元测试 &#x1f6ec; 文章小结&#x1f4d6; 参考资料 &#x1f6eb; 导读 需求 爬虫中经常出现各种类型的命名&#xff0c;往往一个对象…...

RK3568驱动指南|第七篇 设备树-第67章 of操作函数实验:获取属性

瑞芯微RK3568芯片是一款定位中高端的通用型SOC&#xff0c;采用22nm制程工艺&#xff0c;搭载一颗四核Cortex-A55处理器和Mali G52 2EE 图形处理器。RK3568 支持4K 解码和 1080P 编码&#xff0c;支持SATA/PCIE/USB3.0 外围接口。RK3568内置独立NPU&#xff0c;可用于轻量级人工…...

vue3安装vue-router

环境 node 18.14.2 yarn 1.22.19 windows 11 vite快速创建vue项目 参考 安装vue-touter 官网 yarn add vue-router4src下新建router文件夹&#xff0c;该文件夹下新建index.ts // router/index.ts 文件 import { createRouter, createWebHashHistory, RouterOptions, Ro…...

〖大前端 - 基础入门三大核心之JS篇㊱〗- JavaScript 的DOM节点操作

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;不渴望力量的哈士奇(哈哥)&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xf…...

【计算机基础】优雅的PPT就应该这样设计

&#x1f4e2;&#xff1a;如果你也对机器人、人工智能感兴趣&#xff0c;看来我们志同道合✨ &#x1f4e2;&#xff1a;不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 &#x1f4e2;&#xff1a;文章若有幸对你有帮助&#xff0c;可点赞 &#x1f44d;…...

Vatee万腾的科技征程:Vatee数字化创新的前沿探讨

在Vatee万腾的科技征程中&#xff0c;我们目睹了一场数字化创新的引领之旅&#xff0c;探讨了Vatee在科技前沿的独到见解。Vatee万腾不仅仅是一家科技公司&#xff0c;更是一支前行不辍的冒险队伍&#xff0c;通过不断突破自我&#xff0c;探索未知领域&#xff0c;引领着数字化…...

【PB续命05】WinHttp.WinHttpRequest的介绍与使用

0 WinHttp.WinHttpRequest简介 winhttp.winhttprequest是Windows操作系统中的一个API函数&#xff0c;用于创建和发送HTTP请求。它可以用于从Web服务器获取数据&#xff0c;或将数据发送到Web服务器。该函数提供了许多选项&#xff0c;例如设置请求头、设置代理服务器、设置超…...

测试微信模版消息推送

进入“开发接口管理”--“公众平台测试账号”&#xff0c;无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息&#xff1a; 关注测试号&#xff1a;扫二维码关注测试号。 发送模版消息&#xff1a; import requests da…...

镜像里切换为普通用户

如果你登录远程虚拟机默认就是 root 用户&#xff0c;但你不希望用 root 权限运行 ns-3&#xff08;这是对的&#xff0c;ns3 工具会拒绝 root&#xff09;&#xff0c;你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案&#xff1a;创建非 roo…...

Java 加密常用的各种算法及其选择

在数字化时代&#xff0c;数据安全至关重要&#xff0c;Java 作为广泛应用的编程语言&#xff0c;提供了丰富的加密算法来保障数据的保密性、完整性和真实性。了解这些常用加密算法及其适用场景&#xff0c;有助于开发者在不同的业务需求中做出正确的选择。​ 一、对称加密算法…...

Mobile ALOHA全身模仿学习

一、题目 Mobile ALOHA&#xff1a;通过低成本全身远程操作学习双手移动操作 传统模仿学习&#xff08;Imitation Learning&#xff09;缺点&#xff1a;聚焦与桌面操作&#xff0c;缺乏通用任务所需的移动性和灵活性 本论文优点&#xff1a;&#xff08;1&#xff09;在ALOHA…...

基于Java+MySQL实现(GUI)客户管理系统

客户资料管理系统的设计与实现 第一章 需求分析 1.1 需求总体介绍 本项目为了方便维护客户信息为了方便维护客户信息&#xff0c;对客户进行统一管理&#xff0c;可以把所有客户信息录入系统&#xff0c;进行维护和统计功能。可通过文件的方式保存相关录入数据&#xff0c;对…...

LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf

FTP 客服管理系统 实现kefu123登录&#xff0c;不允许匿名访问&#xff0c;kefu只能访问/data/kefu目录&#xff0c;不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...

PostgreSQL——环境搭建

一、Linux # 安装 PostgreSQL 15 仓库 sudo dnf install -y https://download.postgresql.org/pub/repos/yum/reporpms/EL-$(rpm -E %{rhel})-x86_64/pgdg-redhat-repo-latest.noarch.rpm# 安装之前先确认是否已经存在PostgreSQL rpm -qa | grep postgres# 如果存在&#xff0…...

Vue 模板语句的数据来源

&#x1f9e9; Vue 模板语句的数据来源&#xff1a;全方位解析 Vue 模板&#xff08;<template> 部分&#xff09;中的表达式、指令绑定&#xff08;如 v-bind, v-on&#xff09;和插值&#xff08;{{ }}&#xff09;都在一个特定的作用域内求值。这个作用域由当前 组件…...

协议转换利器,profinet转ethercat网关的两大派系,各有千秋

随着工业以太网的发展&#xff0c;其高效、便捷、协议开放、易于冗余等诸多优点&#xff0c;被越来越多的工业现场所采用。西门子SIMATIC S7-1200/1500系列PLC集成有Profinet接口&#xff0c;具有实时性、开放性&#xff0c;使用TCP/IP和IT标准&#xff0c;符合基于工业以太网的…...

13.10 LangGraph多轮对话系统实战:Ollama私有部署+情感识别优化全解析

LangGraph多轮对话系统实战:Ollama私有部署+情感识别优化全解析 LanguageMentor 对话式训练系统架构与实现 关键词:多轮对话系统设计、场景化提示工程、情感识别优化、LangGraph 状态管理、Ollama 私有化部署 1. 对话训练系统技术架构 采用四层架构实现高扩展性的对话训练…...