news 2026/5/1 8:44:32

回溯算法--分割回文串

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
回溯算法--分割回文串

给定一个字符串 s,将 s 分割成一些子串,使每个子串都是回文串。

返回 s 所有可能的分割方案。

示例: 输入: "aab" 输出: [ ["aa","b"], ["a","a","b"] ]

难点

本题的难点在于

  1. 怎么理解或者实现分割这个概念。
  2. 怎么分割的字符串怎么定义。

分割这个问题类似于组合问题,定义一个startIndex,startIndex指向那个字符就是分割的位置。

for(int i = startIndex; i < s.size(); i ++), 在对字符串进行横向遍历的时候,statrIndex 到 i 的长度不就是分割字符串。

思路

定义参数

vector<vector<string>> result; vector<string> path; // 核心的回溯函数 // s: 输入的原始字符串 // startIndex: 当前遍历的起始位置 void backtracking(const string& s, int startIndex)

递归结束条件

当分割到最后一个字符时,说明已经分割完毕,并且一定有满足条件的path,因为每个每个单个字母一定是回文串。

if(startIndex == s.size()){ // 将当前构造的分割方案 (path) 添加到最终结果 (result) 中 result.push_back(path); return; }

单层递归逻辑

首先判断分割的字符串是否为回文串,如果是,则压入path。如果不是则跳过,继续考察更长的子串。

当考察到回文串的时候,才会继续向下递归。

// 单层搜索逻辑:从 startIndex 开始,向后遍历字符串 for(int i = startIndex; i < s.size(); i ++){ // 判断从 startIndex 到 i 的子串是否是回文串 if(isPalindrome(s, startIndex, i)){ // 如果是回文串,将其加入到当前路径 (path) 中 // s.substr(startIndex, i - startIndex + 1) 截取了从 startIndex 开始,长度为 i - startIndex + 1 的子串 path.push_back(s.substr(startIndex, i - startIndex + 1)); }else{ // 如果不是回文串,则跳过,继续考察更长的子串 continue; } // 递归:进入下一层决策,起始位置变为 i + 1 backtracking(s, i + 1); // 回溯:撤销当前层的选择,将刚刚加入的子串弹出,以便探索其他可能性 path.pop_back(); }

代码

#include<iostream> #include<vector> using namespace std; // 解题的核心类 class Solution { private: vector<vector<string>> result; vector<string> path; // 核心的回溯函数 // s: 输入的原始字符串 // startIndex: 当前遍历的起始位置 void backtracking(const string& s, int startIndex){ // 基本情况(终止条件):如果起始位置已经等于字符串长度, // 说明我们已经找到了一个完整的分割方案 if(startIndex == s.size()){ // 将当前构造的分割方案 (path) 添加到最终结果 (result) 中 result.push_back(path); return; } // 单层搜索逻辑:从 startIndex 开始,向后遍历字符串 for(int i = startIndex; i < s.size(); i ++){ // 判断从 startIndex 到 i 的子串是否是回文串 if(isPalindrome(s, startIndex, i)){ // 如果是回文串,将其加入到当前路径 (path) 中 // s.substr(startIndex, i - startIndex + 1) 截取了从 startIndex 开始,长度为 i - startIndex + 1 的子串 path.push_back(s.substr(startIndex, i - startIndex + 1)); }else{ // 如果不是回文串,则跳过,继续考察更长的子串 continue; } // 递归:进入下一层决策,起始位置变为 i + 1 backtracking(s, i + 1); // 回溯:撤销当前层的选择,将刚刚加入的子串弹出,以便探索其他可能性 path.pop_back(); } } bool isPalindrome(const string& s, int i, int j){ while(i <= j){ if(s[i] != s[j]){ return false; }else{ i ++; j --; } } return true; } public: vector<vector<string>> partition(string s) { result.clear(); path.clear(); backtracking(s, 0); return result; } }; int main(){ Solution S; string s = "abcd"; vector<vector<string>> result = S.partition(s); for(auto row : result){ for(auto cols : row){ cout << cols << " "; // 打印子串 } cout << endl; // 每打印完一种方案后换行 } return 0; // 程序正常退出 }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/24 1:10:16

家长用vivo,孩子用iPhone,限制孩子手机使用时长要怎么做?

当家长们深刻地意识到&#xff0c;孩子过度沉迷于手机&#xff0c;可能会对他们的学习产生严重的干扰&#xff0c;导致注意力分散、学习效率低下&#xff1b;在生活方面&#xff0c;也可能使孩子逐渐脱离现实社交&#xff0c;变得孤僻内向&#xff0c;影响他们的人际交往能力。…

作者头像 李华
网站建设 2026/5/1 1:13:08

PostgreSql 常用聚合函数

基础准备 CREATE TEMP TABLE demo(id int, v int, t text); INSERT INTO demo VALUES(1, 10, a),(2, 20, b),(3, NULL, c);idvt110a220b3NULLc注意&#xff1a;版本基于 PG 15&#xff0c;全部自带&#xff0c;无需扩展 A. 基础统计 SELECT count(*) AS cnt, -- 3c…

作者头像 李华
网站建设 2026/4/30 18:08:36

露,信息化集成化信号采集与处理系统信息化集成化信号采集处理系统 机能生理实验系统 信息化集成化生物信号采集与处理系统

一体化生物机能实验系统核心由可移动实验平台、生物信号采集处理系统等四大功能模块构成。该系统将多项实验核心功能与管理模块进行融合&#xff0c;具体涵盖基础实验操作平台、高精度生物信号采集处理单元、附带动物体温实时监测功能的动物呼吸机&#xff0c;以及实验环境监测…

作者头像 李华
网站建设 2026/5/1 7:35:22

文件格式转换神器,牛批了

今天给大家推荐一款非常不错的文件格式转换软件&#xff0c;已经被汉化&#xff0c;没有广高、免费&#xff0c;有需要的小伙伴及时下载收藏。 FileConverter 多媒体格式转换软件 这款工具安装后没有界面&#xff0c;鼠标右键菜单中可以找到&#xff0c;包含了音频、视频、PDF…

作者头像 李华
网站建设 2026/5/1 7:35:46

宽论视角下量化与价值投资的对立统一之道

在金融投资的广阔领域中&#xff0c;宽论犹如一座灯塔&#xff0c;为投资者指引着方向。宽论中对于量化与价值投资对立统一关系的探讨&#xff0c;为我们理解和运用这两种投资策略提供了全新的视角。深入研究宽论中这一核心观点&#xff0c;对于投资者提升投资水平和盈利能力具…

作者头像 李华