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

leetCode 131.分割回文串 + 动态规划 + 回溯算法 + 优化 + 图解 + 笔记

我的往期文章:

leetCode 647.回文子串 动态规划 + 优化空间 / 中心扩展法 + 双指针-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/weixin_41987016/article/details/133883091?spm=1001.2014.3001.5501leetCode 131.分割回文串 + 回溯算法 + 图解 + 笔记-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/weixin_41987016/article/details/134700907?spm=1001.2014.3001.5501(一)利用动态规划来优化判断回文子串

  • 利用动态规划高效地事先一次性计算出, 针对一个字符串s, 它的任何子串是否是回文字串, 然后在我们的回溯函数中直接查询即可, 省去了双指针移动判定这一步骤.(来自代码随想录Carl老师的原话)原文链接:代码随想录 (programmercarl.com)

>>思路和分析

回文子串:讲究的是这个字符串里边左右两边是对称的左右两边的元素是相同的。如果只判断这个字符串的最左面和最右面这两个元素相同的情况下,还知道中间的子串已经是回文的,那么就可以直接判断整个字符串它就是回文子串。

也就是说,如果在[i+1,j-1]范围的子串是一个回文串,再向两边拓展遍历的时候,那只需要判断两边这两个元素是否相同就可以了若相同,dp[i][j]是回文串

>>动规五部曲

1.确定dp数组以及下标的含义

  • dp[i][j]:表示区间范围[i,j]的子串是否为回文子串。如果是,则dp[i][j] = true,否则为false
  • 或者说,dp[i][j] 表示截取从 i 到 j 的子串是否为回文子串

2.确定递推式

if(j == i) dp[i][j]=true;
else if(j-i == 1) dp[i][j] = (s[i]==s[j]);
else dp[i][j] = (s[i] == s[j] && dp[i+1][j-1]);

3.dp 数组初始化

  • dp[i][j]初始化为false

4.确定遍历顺序

一定要从下到上,从左到右遍历,这样能保证dp[i+1][j-1]是经过计算得来的

 5.举例推导dp数组

void computePalindrome(const string& s) {// dp[i][j] 代表s[i:j](双边包括)是否是回文子串dp.resize(s.size(),vector<bool>(s.size(),false));// 根据字符串s,刷新布尔矩阵的大小for(int i=s.size()-1;i>=0;i--) {// 需要倒序计算,保证在i行时,i+1行已经计算好了for(int j=i;j<s.size();j++) {if(j == i) dp[i][j]=true;else if(j-i == 1) dp[i][j] = (s[i]==s[j]);else dp[i][j] = (s[i] == s[j] && dp[i+1][j-1]);}}
}
"aebeaeccfcce"
1  0  0  0  1  0  0  0  0  0  0  0  
0  1  0  1  0  0  0  0  0  0  0  0  
0  0  1  0  0  0  0  0  0  0  0  0  
0  0  0  1  0  1  0  0  0  0  0  0  
0  0  0  0  1  0  0  0  0  0  0  0  
0  0  0  0  0  1  0  0  0  0  0  1  
0  0  0  0  0  0  1  1  0  0  1  0  
0  0  0  0  0  0  0  1  0  1  0  0  
0  0  0  0  0  0  0  0  1  0  0  0  
0  0  0  0  0  0  0  0  0  1  1  0  
0  0  0  0  0  0  0  0  0  0  1  0  
0  0  0  0  0  0  0  0  0  0  0  1  "acgcabbfcc"
1  0  0  0  1  0  0  0  0  0  
0  1  0  1  0  0  0  0  0  0  
0  0  1  0  0  0  0  0  0  0  
0  0  0  1  0  0  0  0  0  0  
0  0  0  0  1  0  0  0  0  0  
0  0  0  0  0  1  1  0  0  0  
0  0  0  0  0  0  1  0  0  0  
0  0  0  0  0  0  0  1  0  0  
0  0  0  0  0  0  0  0  1  1  
0  0  0  0  0  0  0  0  0  1  

(二)分割回文串 + 动态规划 + 回溯算法 + 优化

class Solution {
public:vector<vector<string>> result;vector<string> path; // 放已经回文的子串vector<vector<bool>> dp; // 放事先计算好的是否回文子串的结果void backtracking(const string& s,int startIndex) {// 如果起始位置已经大于 s 的大小,说明已经找到了一组分割方案了if(startIndex >= s.size()) {result.push_back(path);return;}for(int i=startIndex;i<s.size();i++) {if(dp[startIndex][i]) { // 是回文子串// 获取[startIndex,i] 在 s 中的子串string subStr = s.substr(startIndex,i-startIndex+1);path.push_back(subStr);}else continue; // 不是回文,跳过backtracking(s,i+1);// 寻找 i+1 为起始位置的子串path.pop_back();// 回溯过程,弹出本次已经添加的子串}}void computePalindrome(const string& s) {// dp[i][j] 代表s[i:j](双边包括)是否是回文子串dp.resize(s.size(),vector<bool>(s.size(),false));// 根据字符串s,刷新布尔矩阵的大小for(int i=s.size()-1;i>=0;i--) {// 需要倒序计算,保证在i行时,i+1行已经计算好了for(int j=i;j<s.size();j++) {if(j == i) dp[i][j]=true;else if(j-i == 1) dp[i][j] = (s[i]==s[j]);else dp[i][j] = (s[i] == s[j] && dp[i+1][j-1]);}}}vector<vector<string>> partition(string s) {computePalindrome(s);backtracking(s, 0);return result;}
};

参考和推荐文章:

代码随想录 (programmercarl.com)icon-default.png?t=N7T8https://www.programmercarl.com/0131.%E5%88%86%E5%89%B2%E5%9B%9E%E6%96%87%E4%B8%B2.html#%E6%80%9D%E8%B7%AF

摘选代码随想录的总结:

  • 总结难点:
  1. 如何切割?切割问题可以抽象为组合问题
  2. 如何模拟那些切割线?
  3. 切割问题中递归如何终止?
  4. 在递归循环中如何截取子串?
  5. 如何判断回文?

递归用于纵向遍历,for循环用于横向遍历当切割线迭代至字符串末尾,说明找到一种方法。类似组合问题,为了不重复切割同一位置,利用 start_index 作为标记,记录下一轮。递归的起始位置(切割线)。切割过的地方不能重复切割,故递归函数传入 i+1

http://www.lryc.cn/news/250739.html

相关文章:

  • 【傻瓜级JS-DLL-WINCC-PLC交互】3.JS-DLL进行交互
  • 深度学习手势识别算法实现 - opencv python 计算机竞赛
  • 2023-12-01 AIGC-自动生成ppt的AI工具
  • NoSQL 数据建模错误会降低性能
  • 在Android上搭建一个NDK项目
  • TOP-K问题和向上调整算法和向下调整算法的时间复杂度问题的分析
  • 3、服务器性能剖析
  • xxl-job 分布式任务调度框架
  • 软件使用-stm32入门
  • 使用MAT分析内存泄漏(mac)
  • 【Vue】Linux 运行 npm run serve 报错 vue-cli-service: Permission denied
  • LeetCode的几道题
  • NLP/Natural Language Processing
  • 【教学类-06-12】20231202 0-9数字分合-房屋样式(一)-下右空-升序-抽7题
  • uni-app 微信小程序 电子签名及签名图片翻转显示功能
  • MySQL 8.0关键字和保留字
  • PyLMKit(3):基于角色扮演的应用案例
  • JAVA全栈开发 集合详解(day14+day15汇总)
  • Linux Spug自动化运维平台本地部署与公网远程访问
  • zookeeper集群和kafka集群
  • Java——》JSONObjet 数据顺序
  • 【个人记录】NGINX反向代理grpc服务
  • 【小白推荐】安装OpenCV4.8 系统 Ubuntu 22.04LST Linux.
  • 使用Docker Compose搭建CIG监控平台
  • 前端文本省略号后面添加复制文字
  • 【算法】动态规划中的路径问题
  • 代数学笔记9: 群的直积,可解群,自由群,群表示
  • kali学习
  • 《论文阅读》DualGATs:用于对话中情绪识别的双图注意力网络
  • 【算法】单调栈题单——字典序最小⭐(一种类型的模板题)