news 2026/7/2 1:22:22

回溯算法--递增子序列

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
回溯算法--递增子序列
输入:nums = [4,4,3,2,1]输出:[[4,4]]

注意点

  1. 此题目的集合是无序的,并且要求同一层之间的去重,因此和之前有序的同一层去重(used数组)不同,千万不能混淆。
  2. 此题还需要对保证输出的组合是有序的,因此怎么保证path是有序的。

思路

  1. 无序集合的树层之间去重,可以使用unordered_set,记录每一层出现过的元素,在for循环之前定义,一个for循环是一层,因此要在for循环之前定义。并且每一层都单独需要一个unorered_set来记录每一层是否重复,因此不需要对unordered_set进行回溯。
  2. 要保证有序,就是要保证正在访问的元素nums[i] > path数组中最后一个元素,path.back可以表示最后一个元素。但是使用back要保证nums数组不能为空。

代码

回溯三部曲,

参数

void backtracking(const vector<int>& nums, int startIndex)

终止条件

其实也可以不需要终止条件,因为递归会一直遍历,一直寻找合适的path,即走完所有的for循环自动停止。

if (path.size() > 1) { result.push_back(path); } // 终止条件2:如果路径长度等于原数组长度,不再继续(虽然这种情况很少) if (path.size() == nums.size()) return;

单层循环逻辑

  • 为什么unordered_set创建的位置在for循环之前
  • 为什么unordered_set不需要回溯
  • nums.back使用的前提
  • 为什么if条件里面的剪枝操作是或的关系
  • 为什么是continue而不是break
// 关键:unordered_set用于记录本层元素是否重复使用 // 注意:这个uset的生命周期只在本层递归中,每次进入新的递归层都会重新定义 unordered_set<int> uset; // 遍历从startIndex开始的所有可能选择 for (int i = startIndex; i < nums.size(); i ++) { // 剪枝条件1:如果当前元素小于路径最后一个元素,跳过(不满足递增) // 注意:需要先检查path是否为空,否则path.back()会出错 // 剪枝条件2:如果当前元素在本层已经使用过,跳过(去重) // 注意:这里的去重是针对同一递归层,不是针对整个递归树 if ((!path.empty() && nums[i] < path.back()) || uset.find(nums[i]) != uset.end()) continue; uset.insert(nums[i]); path.push_back(nums[i]); // 递归:从i+1开始继续寻找(注意是i+1,不是i,因为不能重复使用同一索引的元素) backtracking(nums, i + 1); path.pop_back(); // 注意:uset不需要撤销,因为它在栈上,每次递归会重新创建 }

整体代码

class Solution { private: vector<vector<int>> result; // 存储所有递增子序列的结果 vector<int> path; // 存储当前正在构建的递增子序列 // 回溯函数:寻找所有递增子序列 // nums: 输入数组 // startIndex: 当前递归开始选择的起始索引 void backtracking(const vector<int>& nums, int startIndex) { // 终止条件1:当路径长度大于等于2时,保存当前递增子序列 // 题目要求子序列长度至少为2 if (path.size() > 1) { result.push_back(path); } // 终止条件2:如果路径长度等于原数组长度,不再继续(虽然这种情况很少) if (path.size() == nums.size()) return; // 关键:unordered_set用于记录本层元素是否重复使用 // 注意:这个uset的生命周期只在本层递归中,每次进入新的递归层都会重新定义 unordered_set<int> uset; // 遍历从startIndex开始的所有可能选择 for (int i = startIndex; i < nums.size(); i ++) { // 剪枝条件1:如果当前元素小于路径最后一个元素,跳过(不满足递增) // 注意:需要先检查path是否为空,否则path.back()会出错 // 剪枝条件2:如果当前元素在本层已经使用过,跳过(去重) // 注意:这里的去重是针对同一递归层,不是针对整个递归树 if ((!path.empty() && nums[i] < path.back()) || uset.find(nums[i]) != uset.end()) continue; uset.insert(nums[i]); path.push_back(nums[i]); // 递归:从i+1开始继续寻找(注意是i+1,不是i,因为不能重复使用同一索引的元素) backtracking(nums, i + 1); path.pop_back(); // 注意:uset不需要撤销,因为它在栈上,每次递归会重新创建 } } public: vector<vector<int>> findSubsequences(vector<int>& nums) { result.clear(); path.clear(); backtracking(nums, 0); return result; } };
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/29 21:00:22

AI助力WSL2安装Ubuntu:自动解决常见错误

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个WSL2安装Ubuntu的AI辅助工具&#xff0c;要求实现以下功能&#xff1a;1.自动检测Windows系统版本和WSL2支持状态 2.智能选择最适合的Ubuntu版本 3.自动处理安装过程中的常…

作者头像 李华
网站建设 2026/6/25 7:56:07

AI如何优化Apache Airflow工作流编排

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个基于Apache Airflow的智能工作流编排系统&#xff0c;集成AI模型来自动生成优化的DAG结构&#xff0c;根据历史执行数据预测任务执行时间并动态调整调度策略。系统应包含任…

作者头像 李华
网站建设 2026/6/28 21:16:31

SpringBoot+MyBatisPlus入门:10分钟搭建第一个应用

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个最简单的SpringBootMyBatisPlus入门示例&#xff0c;要求&#xff1a;1. 使用Spring Initializr创建项目&#xff1b;2. 添加MyBatisPlus依赖&#xff1b;3. 创建Student实…

作者头像 李华
网站建设 2026/6/28 20:37:04

2、深入探索Shell输入、输出与吞吐量

深入探索Shell输入、输出与吞吐量 1. 位置参数与特殊参数 在Shell脚本中,位置参数是非常重要的概念。Bourne shell 只能处理最多 9 个位置参数,如果脚本中使用 $10 ,它会被解释为 $1 后面跟着一个零。为了兼容旧脚本,bash 保留了这种行为。若要访问大于 9 的位置参数…

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

nVisual拓扑、设备、业务关联映射监测示例

nVisual拓扑监测支持网元对象与机房机柜物理设备、业务流程图进行关联映射&#xff0c;全位展示监测对象的关系视图&#xff0c;提高故障排查效率与业务保障能力。 1、网元对象实时显示对象online/offline状态&#xff1b; 2、选中网元图标&#xff0c;点击右侧“监测”&#…

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

Docker网络架构深度解析:从原理到实战

前言 Docker 容器技术的普及改变了软件交付的方式&#xff0c;而网络作为容器化应用交互的基础设施&#xff0c;其重要性不言而喻。Docker 提供了多种网络驱动&#xff0c;以适应不同的应用场景。本文将深入剖析 Docker 的核心网络模式&#xff0c;包括 Bridge&#xff08;桥接…

作者头像 李华