递归 回溯算法详解

什么是回溯算法

回溯算法是⼀种经典的递归算法,通常⽤于解决组合问题、排列问题和搜索问题等。
回溯算法的基本思想:从⼀个初始状态开始,按照⼀定的规则向前搜索,当搜索到某个状态⽆法前进
时,回退到前⼀个状态,再按照其他的规则搜索。回溯算法在搜索过程中维护⼀个状态树,通过遍历 状态树来实现对所有可能解的搜索。
回溯算法的核⼼思想:“试错”,即在搜索过程中不断地做出选择,如果选择正确,则继续向前搜
索;否则,回退到上⼀个状态,重新做出选择。回溯算法通常⽤于解决具有多个解,且每个解都需要 搜索才能找到的问题。

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

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

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

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

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

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

总结 回溯算法是⼀种⾮常重要的算法,可以解决许多组合问题、排列问题和搜索问题等。回溯算法的核⼼
思想是搜索状态树,通过遍历状态树来实现对所有可能解的搜索。回溯算法的模板⾮常简单,但是实
现起来需要注意⼀些细节,⽐如如何做出选择、如何撤销选择等。

题目一: 全排列

1. 题⽬链接:

https://leetcode.cn/problems/permutations/description/

2. 题⽬描述:

在这里插入图片描述

3. 解法:

算法思路:

的回溯题⽬,我们需要在每⼀个位置上考虑所有的可能情况并且不能出现重复。通过深度优先搜索的⽅式,不断地枚举每个数在当前位置的可能性,并回溯到上⼀个状态,直到枚举完所有可能性,得到正确的结果。 每个数是否可以放⼊当前位置,只需要判断这个数在之前是否出现即可。

递归流程如下:
  1. ⾸先定义⼀个List<List<Integer>> ret⽤来存放所有可能的排列,⼀个 List<Integer> path⽤来存放每个状态的排列,⼀个⼀维数组boolean[] check标记元素,然后从第⼀个位置开始进⾏递归;

  2. 递归结束条件:当path.size()等于 nums 数组的⻓度时,说明我们已经处理完了所有数字,将当前数组存⼊结果中;

  3. 在每个递归状态中,枚举所有下标 i,若这个下标未被标记,即check[i]==false,则使⽤ nums 数组中当前下标的元素:

    a. 将 check[i] 标记为 true;

    b. path中添加nums[i]

    c. 进入下一层递归

    d. 恢复现场,将 check[i] 标记为 false,表⽰回溯;

  4. 最后,返回 ret。

4.代码

class Solution 
{

 List<List<Integer>> ret;
 List<Integer> path;
 boolean[] check;
 public List<List<Integer>> permute(int[] nums) 
 {
 ret = new ArrayList<>();
 path = new ArrayList<>();
 check = new boolean[nums.length];
 dfs(nums);
 return ret;
 }
 public void dfs(int[] nums)
 {
 if(nums.length == path.size())
 {
 ret.add(new ArrayList<>(path));
 return;
 }
 for(int i = 0; i < nums.length; i++)
 {
 if(check[i] == false)
 {
 path.add(nums[i]);
 check[i] = true;
 dfs(nums);
 // 回溯 -> 恢复现场 
 check[i] = false;
 path.remove(path.size() - 1);
 }
 }
 }
}

题目二:⼦集

1. 题⽬链接:

https://leetcode.cn/problems/subsets/description/

2. 题目描述:

在这里插入图片描述

3. 解法:

算法思路:

为了获得 nums 数组的所有⼦集,我们需要对数组中的每个元素进⾏选择或不选择的操作,即 nums 数组⼀定存在 2^(数组⻓度)
个⼦集。对于查找⼦集,具体可以定义⼀个数组,来记录当前的状态,并 对其进⾏递归。

对于每个元素有两种选择:1. 不进⾏任何操作;
2. 将其添加⾄当前状态的集合。在递归时我们需要保
证递归结束时当前的状态与进⾏递归操作前的状态不变,⽽当我们在选择进⾏步骤 2 进⾏递归时,当
前状态会发⽣变化,因此我们需要在递归结束时撤回添加操作,即进⾏回溯。
递归函数设计:
public void dfs(int[] nums,int pos)
参数:pos(当前需要处理的元素下标);
返回值:⽆;
函数作⽤:查找集合的所有⼦集并存储在答案列表中。

递归流程如下:
1.递归结束条件:如果当前需要处理的元素下标越界,则记录当前状态并直接返回;
2. 在递归过程中,对于每个元素,我们有两种选择:

(1)不选择当前元素,直接递归到下⼀个元素;

(2)选择当前元素,将其添加到数组末尾后递归到下⼀个元素,然后在递归结束时撤回添加操作;
3. 所有符合条件的状态都被记录下来,返回即可。

4.代码

// 解法⼀: 

class Solution 
{
 List<List<Integer>> ret;
 List<Integer> path;
 public List<List<Integer>> subsets(int[] nums) 
 {
 ret = new ArrayList<>();
 path = new ArrayList<>();
 dfs(nums, 0);
 return ret;
 }

 public void dfs(int[] nums, int pos)
 {
 if(pos == nums.length)
 {
 ret.add(new ArrayList<>(path));
 return;
 }
 // 选 
 path.add(nums[pos]);
 dfs(nums, pos + 1);
 path.remove(path.size() - 1); // 恢复现场 
 // 不选 
 dfs(nums, pos + 1);
 }
}

// 解法⼆: 

class Solution 
{
 List<List<Integer>> ret;
 List<Integer> path;
 public List<List<Integer>> subsets(int[] nums) 
 {
 ret = new ArrayList<>();
 path = new ArrayList<>();
 dfs(nums, 0);
 return ret;
 }
 public void dfs(int[] nums, int pos)
 {
 ret.add(new ArrayList<>(path));
 for(int i = pos; i < nums.length; i++)
 {
 path.add(nums[i]);
 dfs(nums, i + 1);
 path.remove(path.size() - 1); // 恢复现场 
 }
 }
}
### 解析 Import Error 的常见原因 当遇到 `ImportError: cannot import name 'Generic'` 错误时,通常意味着尝试从模块中导入的对象不存在或无法访问。此问题可能由多种因素引起: - 版本不兼容:不同库之间的版本冲突可能导致此类错误。 - 安装缺失:目标库未正确安装或路径配置有误。 - 导入语句不当:可能存在循环依赖或其他语法层面的问题。 ### 针对 Generic 类型的具体解决方案 对于特定于 `Generic` 的情况,考虑到 Python 中 `Generic` 是 typing 模块的一部分,在处理该类别的 ImportError 时可采取如下措施[^1]: #### 方法一:确认typing模块可用性 确保环境中已安装标准库中的 typing 模块,并且其版本支持所使用的特性。可以通过以下命令验证: ```bash python -c "from typing import Generic; print(Generic)" ``` 如果上述命令执行失败,则可能是由于 Python 或者相关扩展包的版本过低造成的。此时应考虑升级至更高版本的解释器以及对应的开发工具链。 #### 方法二:调整导入方式 有时直接通过顶层命名空间来获取所需组件会更稳定可靠。修改代码以采用这种做法可能会解决问题: ```python from collections.abc import Iterable # 如果是迭代器相关接口 from typing import TypeVar, Protocol # 对于协议和泛型定义 T = TypeVar('T') class MyContainer(Protocol[T]): ... ``` 注意这里并没有显式提到 `Generic` ,而是利用了更为基础的数据结构抽象基类或是其他替代方案实现相同功能[^2]。 #### 方法三:排查环境变量设置 检查系统的 PYTHONPATH 和虚拟环境配置是否正常工作。任何异常都可能导致某些第三方软件包找不到必要的资源文件而引发类似的错误提示。建议清理并重建项目专属的工作区以便排除干扰项的影响。 #### 示例修正后的代码片段 假设原始代码试图这样引入 `Generic` : ```python from some_module import Generic # 可能导致 ImportError ``` 改为遵循官方文档推荐的方式后变为: ```python from typing import Generic # 正确的做法 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值