【回溯+剪枝】回溯算法的概念 全排列问题

server/2025/2/3 2:56:50/

文章目录


在这里插入图片描述

46. 全排列

46. 全排列

​ 给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。

示例 1:

输入:nums = [1,2,3]
输出:[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]]

示例 2:

输入:nums = [0,1]
输出:[[0,1],[1,0]]

示例 3:

输入:nums = [1]
输出:[[1]]

提示:

  • 1 <= nums.length <= 6
  • -10 <= nums[i] <= 10
  • nums 中的所有整数 互不相同

Ⅰ. 什么是回溯算法❓❓❓

回溯算法是一种经典的递归算法,通常用于解决组合问题、排列问题和搜索问题等。

回溯算法的基本思想:从一个初始状态开始,按照一定的规则向前搜索,当搜索到某个状态无法前进时,回退到前一个状态,再按照其他的规则搜索。回溯算法在搜索过程中维护一个状态树,通过遍历决策树来实现对所有可能解的搜索。

回溯算法的核心思想:“试错”,即在搜索过程中不断地做出选择,如果选择正确,则继续向前搜索;否则,回退到上一个状态,重新做出选择。回溯算法通常用于解决具有多个解,且每个解都需要搜索才能找到的问题,也就是相当于是 枚举

​ 其实说白了,回溯就是深搜,只不过回溯更加强调返回操作对一些题的理解是很重要的,所以专门给这种理解起了个回溯的专用词!其实回溯是有模板的,但是给出模板之后反而会限制思维,会想着怎么根据模板出发,而不是从题目本身出发,这是不行的,所以这里并不提供什么模板去背诵,当我们真正理解了回溯之后,其实写出来代码就会发现是和模板类似的!

Ⅱ. 回溯算法的应用

1、组合问题

​ 组合问题是指从给定的⼀组数(不重复)中选取出所有可能的 k 个数的组合。例如,给定数集 [1,2,3],要求选取 k=2 个数的所有组合。其结果为:

[1, 2]
[1, 3]
[2, 3]

2、排列问题

​ 排列问题是指从给定的⼀组数(可重复)中选取出所有可能的 k 个数的排列。例如,给定数集 [1,2,3],要求选取 k=2 个数的所有排列。其结果为:

[1,2]
[2,1]
[1,3]
[3,1]
[2,3]
[3,2]

3、子集问题

​ 子集问题是指从给定的一组数中选取出所有可能的子集,其中每个子集中的元素可以按照任意顺序排列。例如,给定数集 [1,2,3],要求选取所有可能的子集。其结果为:

Ⅲ. 解题思路:回溯 + 剪枝

​ 首先根据题目要求,我们 需要两个全局变量 retpathret 就是我们最后要返回的结果集,而 path 是存放当前正在走的全排列的元素!为什么要将它们设置为全局变量,而不是在函数中作为引用传递,其实是简化了函数需要填充的内容,并且在一定程度上也简化了我们的操作,尤其是后面一些题目比较说 N皇后的题目等等,如果用传引用的方式的话,其实是不太好控制的!虽说使用全局变量之后,回溯的时候需要我们手动去恢复一下 path 的状态,但是这都是值得的!

​ 对于这类排列组合的问题,我们最好是能画出一棵详细一点的决策树(不要想的高大上,其实就是把情况枚举出来的树而已),这样子有利于帮助我们理解其中要注意的细节,如下面的图片所示!

​ 因为排列问题需要遍历所有的组合可能,包括顺序不同的组合可能,比如说 [1, 2][2, 1] 都需要满足,所以我们在排列问题中就不需要使用 index 来控制每次下一层也就是树枝之间的起始位置,只需要 每次都从数组下标为 0 开始遍历所有可能即可

在这里插入图片描述

​ 但是这里还有一个问题,就是可能会出现重复调用了某个 nums[i],比如说 [1, 2, 3] 中我们如果不对每个元素进行控制的话,可能会排列出 [1, 1, 1] 等情况,但是排列问题中我们 只能让每个元素都出现一次,所以需要使用一个 used 数组进行判断

  1. used 数组初始化为 false
  2. 因为会出现重复的情况只在一条路径上,所以我们无需担心路径之间的重复,所以我们让 used[i]false 表示是 nums[i] 还没走过,当我们在递归这条路径的下一个节点之前将 used[i] 变成 true,此时下一个节点层就会知道该 nums[i] 是走过的了,就不会再走了!然后回溯的时候将 used[i] 变成 false,这样子对同一树层是没有影响的,只会对下一层的路径产生影响!
  3. 简单地说,当我们 判断到 used[i] == true,也就说明上一层已经将这个元素遍历过了,所以我们直接 continue 即可

​ 说白了,上面的操作其实就是一个 剪枝 的操作!

​ 对于递归函数出口,我们只需要判断一下 path 数组的长度,是不是等于题目给的 nums 的长度,是的话说明当前序列已经是完成的了,则添加到 ret 结果集中然后直接返回即可!

​ 然后就是递归函数的主体,其实最重要的就是三步:

  1. 处理当前层节点(处理)
  2. 递归下一层节点(递归)
  3. 进行回溯后的处理,防止影响后面的结果(回溯

​ 可以结合下面的代码一起分析这个过程!

class Solution {
private:vector<vector<int>> ret; // 结果集vector<int> path;        // 存放当前正在走的全排列的元素vector<bool> used;       // 标记哪个元素已经走过了,用于剪枝操作,防止重复
public:vector<vector<int>> permute(vector<int>& nums) {used.resize(nums.size(), false);dfs(nums);return ret;}void dfs(vector<int>& nums){// 递归函数出口(将当前path添加到结果集中)if(path.size() == nums.size()){ret.push_back(path);return;}for(int i = 0; i < nums.size(); ++i) // 每次都是从0开始遍历,和组合问题不一样,排列问题需要遍历所有可能{// 进行剪枝操作,防止结果重复if(used[i] == true)continue;// 处理当前层节点path.push_back(nums[i]);used[i] = true;// 递归下一层节点dfs(nums);// 进行回溯后的处理,防止影响后面的结果path.pop_back();used[i] = false;}}
};

在这里插入图片描述


http://www.ppmy.cn/server/164499.html

相关文章

网站结构优化:加速搜索引擎收录的关键

本文来自&#xff1a;百万收录网 原文链接&#xff1a;https://www.baiwanshoulu.com/9.html 网站结构优化对于加速搜索引擎收录至关重要。以下是一些关键策略&#xff0c;旨在通过优化网站结构来提高搜索引擎的抓取效率和收录速度&#xff1a; 一、合理规划网站架构 采用扁…

什么是Rust?它有什么特点?为什么要学习Rust?

什么是Rust&#xff1f;它有什么特点&#xff1f;为什么要学习Rust&#xff1f; 如果你是一名编程初学者&#xff0c;或者已经有一些编程经验但对Rust感兴趣&#xff0c;那么这篇文章就是为你准备的&#xff01;我们将用简单易懂的语言&#xff0c;带你了解Rust是什么、它有什…

Linux中 端口被占用如何解决

lsof命令查找 查找被占用端口 lsof -i :端口号 #示例 lsof -i :8080 lsof -i :3306 netstat命令查找 查找被占用端口 netstat -tuln | grep 端口号 #示例 netstat -tuln | grep 3306 netstat -tuln | grep 6379 ss命令查找 查找被占用端口 ss -tunlp | grep 端口号 #示例…

Cyber Security 101-Build Your Cyber Security Career-Security Principles(安全原则)

了解安全三元组以及常见的安全模型和原则。 任务1&#xff1a;介绍 安全已成为一个流行词;每家公司都想声称其产品或服务是安全的。但事实真的如此吗&#xff1f; 在我们开始讨论不同的安全原则之前&#xff0c;了解我们正在保护资产的对手至关重要。您是否试图阻止蹒跚学步…

A4988一款常用的步进电机驱动芯片

A4988 是一款常用的步进电机驱动芯片&#xff0c;广泛应用于 3D 打印机、CNC 机床和小型自动化设备中。它可以驱动多种类型的步进电机&#xff0c;但需要根据电机的参数&#xff08;如电压、电流、相数等&#xff09;进行合理配置。 一、A4988 的主要特性 驱动能力&#xff1a;…

如何获取Springboot项目运行路径 (idea 启动以及打包为jar均可) 针对无服务器容器新建上传文件路径(适用于win 与 linunix)

public class Constants {public static String getUploadDir() {// 获取 JAR 包所在目录ApplicationHome home new ApplicationHome(Constants.class);File jarDir home.getDir();// 构建上传文件存储路径&#xff08;JAR 同级目录下的 uploads 文件夹&#xff09;File uplo…

.Net / C# 繁体中文 与 简体中文 互相转换, 支持地方特色词汇

版本号 Nuget 搜索 “OpenCCNET”, 注意别找错, 好多库的名字都差不多 支持 “繁,简” 的互相转换, 支持多个地区常用词汇的转换, 还支持 日文的新旧转换. OpenCC 在 .Net 中的实现 https://github.com/CosineG/OpenCC.NET <PackageReference Include"OpenCCNET"…

分布式微服务系统架构第88集:kafka集群

使用集 群最大的好处是可以跨服务器进行负载均衡&#xff0c;再则就是可以使用复制功能来避免因单点故 障造成的数据丢失。在维护 Kafka 或底层系统时&#xff0c;使用集群可以确保为客户端提供高可用 性。 需要多少个broker 一个 Kafka 集群需要多少个 broker 取决于以下几个因…