回溯算法--递增子序列

复制代码
输入: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数组不能为空。

代码

回溯三部曲,

参数

cpp 复制代码
void backtracking(const vector<int>& nums, int startIndex)

终止条件

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

cpp 复制代码
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
cpp 复制代码
	// 关键: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不需要撤销,因为它在栈上,每次递归会重新创建
		}

整体代码

cpp 复制代码
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;     
	}
};
相关推荐
m0_639397297 小时前
代码随想录算法训练营第五十天|图论理论基础,深搜理论基础,98. 所有可达路径,广搜理论基础
算法·图论
艾莉丝努力练剑7 小时前
【Python库和代码案例:第一课】Python 标准库与第三方库实战指南:从日期处理到 Excel 操作
java·服务器·开发语言·人工智能·python·pycharm·pip
yugi9878387 小时前
基于C#实现的WiFi信号强度扫描程序
开发语言·c#
乂爻yiyao7 小时前
Java 的云原生困局与破局
java·开发语言·云原生
鸿儒5177 小时前
记录一个C++操作8位影像的一个bug
开发语言·c++·bug
脏脏a7 小时前
深度剖析 C++ string:从 0 到 1 的模拟实现与细节解析
开发语言·c++
创作者mateo7 小时前
python基础学习之Python 循环及函数
开发语言·python·学习
智驱力人工智能7 小时前
无人机车辆密度检测系统价格 询价准备 需要明确哪些参数 物流园区无人机车辆调度系统 无人机多模态车流密度检测技术
深度学习·算法·安全·yolo·无人机·边缘计算
福尔摩斯张7 小时前
【实战】C/C++ 实现 PC 热点(手动开启)+ 手机 UDP 自动发现 + TCP 通信全流程(超详细)
linux·c语言·c++·tcp/ip·算法·智能手机·udp