news 2026/6/9 22:08:09

回溯算法--分割回文串

作者头像

张小明

前端开发工程师

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/6/9 19:03:13

君耀压敏电阻25D系列的高性能过压保护解决方案

君耀压敏电阻25D系列是君耀&#xff08;Brightking&#xff09;推出的高性能压敏电阻产品&#xff0c;广泛应用于电子电路的过压保护。该系列产品以其快速响应、高浪涌电流承受能力以及广泛的电压范围而备受市场青睐。可以说&#xff0c;君耀压敏电阻25D系列是您的高性能过压保…

作者头像 李华
网站建设 2026/6/9 19:23:40

STM32 SAI接口TDM模式深度解析:多通道音频传输的终极解决方案

STM32 SAI接口TDM模式深度解析&#xff1a;多通道音频传输的终极解决方案在音频处理和多通道数据采集领域&#xff0c;TDM&#xff08;时分复用&#xff09;技术是解决高通道数传输的关键。本文将深入探讨STM32 SAI接口的TDM模式&#xff0c;揭示其在专业音频系统、多通道数据采…

作者头像 李华
网站建设 2026/6/10 12:26:27

市面上沙特二手车清关证书哪家好

市面上沙特二手车清关证书哪家好&#xff1f;随着沙特进口二手车市场需求的日益增大&#xff0c;二手汽车清关成为了众多从业者关注的重点&#xff0c;而获得符合要求的清关证书是顺利清关的关键。在市面上众多提供沙特二手车清关证书服务的机构中&#xff0c;ASTC-TI那尔泰柯尼…

作者头像 李华
网站建设 2026/6/10 12:28:55

AI大模型优化了谁?程序员还是产品经理?

一名失业中的程序员&#xff0c;因一次创新的求职方式引发全网关注。林默然以999元的价格&#xff0c;在上海地铁陆家嘴站租用广告位5天&#xff0c;展示个人简历二维码&#xff0c;扫码即可追溯其"辗转的职场生涯"。 林默然于2023年5月选择"主动离职"。同…

作者头像 李华
网站建设 2026/6/10 1:52:54

电商客服机器人背后的技术支柱:Qwen3-14B实战

电商客服机器人背后的技术支柱&#xff1a;Qwen3-14B实战 在电商平台日均处理数百万用户咨询的今天&#xff0c;一个“能说会做”的智能客服系统早已不再是锦上添花的功能&#xff0c;而是保障用户体验和运营效率的核心基础设施。然而&#xff0c;许多企业尝试引入大模型时却陷…

作者头像 李华
网站建设 2026/6/10 12:33:39

Foundation 按钮组

Foundation 5 按钮组&#xff08;Button Groups&#xff09; Foundation 5 的按钮组用于将多个按钮紧密组合在一起&#xff0c;形成统一的视觉效果。支持颜色统一、大小一致、圆角、平均分配宽度&#xff0c;以及堆叠&#xff08;小屏自动垂直排列&#xff09;等功能。非常适合…

作者头像 李华