搜索
查看
编辑修改
首页
UNITY
NODEJS
PYTHON
AI
GIT
PHP
GO
CEF3
JAVA
HTML
CSS
搜索
羊村懒王
这个屌丝很懒,什么也没留下!
关注作者
热门标签
jquery
HTML
CSS
PHP
ASP
PYTHON
GO
AI
C
C++
C#
PHOTOSHOP
UNITY
iOS
android
vue
xml
爬虫
SEO
LINUX
WINDOWS
JAVA
MFC
CEF3
CAD
NODEJS
GIT
Pyppeteer
article
热门文章
1
Kafka 集群配置SASL+ACL_unexpected handshake request with client mechanism
2
养老服务平台市场现状研究分析-_养老服务平台市场分析
3
Python故障诊断与异常检测_故障诊断与python学习 csdn
4
【数据结构】顺序表和单链表的定义与一些基本操作_顺序表和单链表的类型声明
5
【无标题】Windows10安全中心永久关闭教程_win10关闭安全中心 csdn
6
Matlab导入数据绘制图像_在matlab中导入excel绘制曲线图
7
目标检测5:采用yolov8, RK3568上推理实时视频流_yolov8 转rk3568
8
2020国考判断推理高频考点三段论前提论证型_判断推理三段论
9
【leetcode面试经典150题】46. 存在重复元素 II(C++)
10
回归预测 | Matlab实现SSA-GRNN麻雀算法优化广义回归神经网络多变量回归预测(含优化前后预测可视化)
当前位置:
article
> 正文
递归、搜索与回溯算法:回溯,决策树
作者:羊村懒王 | 2024-04-23 15:08:39
赞
踩
递归、搜索与回溯算法:回溯,决策树
回溯算法是⼀种经典的递归算法,通常⽤于解决组合问题、排列问题和搜索问题等。
回溯算法的基本思想:从⼀个初始状态开始,按照⼀定的规则向前搜索,当搜索到某个状态⽆法前进时,回退到前⼀个状态,再按照其他的规则搜索。回溯算法在搜索过程中维护⼀个状态树,通过遍历状态树来实现对所有可能解的搜索。
回溯算法的核⼼思想:“试错”,即在搜索过程中不断地做出选择,如果选择正确,则继续向前搜
索;否则,回退到上⼀个状态,重新做出选择。回溯算法通常⽤于解决具有多个解,且每个解都需要搜索才能找到的问题。
1.回溯算法的模板
void
backtrack
(vector<
int
>& path, vector<
int
>& choice, ...) {
//
满⾜结束条件
if
(
/*
满⾜结束条件
*/
) {
//
将路径添加到结果集中
ret.
push_back
(path);
return
;
}
//
遍历所有选择
for
(
int
i =
0
; i < choices.
size
(); i++) {
//
做出选择
path.
push_back
(choices[i]);
//
做出当前选择后继续搜索
backtrack
(path, choices);
//
撤销选择
path.
pop_back
();
}
}
其中,
path
表⽰当前已经做出的选择,
choices
表⽰当前可以做的选择。在回溯算法中,我们需
要做出选择,然后递归地调⽤回溯函数。如果满⾜结束条件,则将当前路径添加到结果集中;否则,我们需要撤销选择,回到上⼀个状态,然后继续搜索其他的选择。
回溯算法的时间复杂度通常较⾼,因为它需要遍历所有可能的解。但是,回溯算法的空间复杂度较
低,因为它只需要维护⼀个状态树。在实际应⽤中,回溯算法通常需要通过剪枝等⽅法进⾏优化,以减少搜索的次数,从⽽提⾼算法的效率。
2.回溯算法的应⽤
组合问题
组合问题是指从给定的⼀组数(不重复)中选取出所有可能的 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
]
总结
回溯算法是⼀种⾮常重要的算法,可以解决许多组合问题、排列问题和搜索问题等。回溯算法的核⼼思想是搜索状态树,通过遍历状态树来实现对所有可能解的搜索。回溯算法的模板⾮常简单,但是实现起来需要注意⼀些细节,⽐如如何做出选择、如何撤销选择等。
例题一
解法:
算法思路:
典型的回溯题⽬,我们需要在每⼀个位置上考虑所有的可能情况并且不能出现重复。通过深度优先搜索的⽅式,不断地枚举每个数在当前位置的可能性,并回溯到上⼀个状态,直到枚举完所有可能性,得到正确的结果。
每个数是否可以放⼊当前位置,只需要判断这个数在之前是否出现即可。具体地,在这道题⽬中,我们可以通过⼀个递归函数 dfs 和标记数组 check来实现全排列。
例题二
例题三
例题四
解法:
算法思路:
因为题⽬不要求返回的排列顺序,因此我们可以对初始状态排序,将所有相同的元素放在各⾃相邻的位置,⽅便之后操作。因为重复元素的存在,我们在选择元素进⾏全排列时,可能会存在重复排列,例如:[1, 2, 1],所有的 下标排列 为:
123 132 213 231 312 321
按照以上下标进⾏排列的结果为:
121 112 211 211 112 121
可以看到,有效排列只有三种[1, 1, 2],[1, 2, 1],[2, 1, 1],其中每个排列都出现两次。因此,我们需要对相同元素定义⼀种规则,使得其组成的排列不会形成重复的情况:
1.
我们可以将相同的元素按照排序后的下标顺序出现在排列中,通俗来讲,若元素 s 出现 x 次,则排序后的第 2 个元素 s ⼀定出现在第 1 个元素 s 后⾯,排序后的第 3 个元素 s ⼀定出现在第 2 个元素 s 后⾯,以此类推,此时的全排列⼀定不会出现重复结果。
2.
例如:a1=1,a2=1,a3=2,排列结果为 [1, 1, 2] 的情况只有⼀次,即 a1 在 a2 前⾯,因为 a2 不会出现在 a1 前⾯从⽽避免了重复排列。
3.
我们在每⼀个位置上考虑所有的可能情况并且不出现重复;
4.
*注意*:若当前元素的前⼀个相同元素未出现在当前状态中,则当前元素也不能直接放⼊当前状态的数组,此做法可以保证相同元素的排列顺序与排序后的相同元素的顺序相同,即避免了重复排列出现。
5.
通过深度优先搜索的⽅式,不断地枚举每个数在当前位置的可能性,并在递归结束时回溯到上⼀个状态,直到枚举完所有可能性,得到正确的结果。
递归函数设计:void backtrack(vector<int>& nums, int pos)
参数:pos(当前需要填⼊的位置);
返回值:⽆;
函数作⽤:查找所有合理的排列并存储在答案列表中。
递归流程如下:
1.
定义⼀个⼆维数组 ret ⽤来存放所有可能的排列,⼀个⼀维数组 path ⽤来存放每个状态的排列,⼀个⼀维数组 check 标记元素,然后从第⼀个位置开始进⾏递归;
2.
在每个递归的状态中,我们维护⼀个步数 pos,表⽰当前已经处理了⼏个数字;
3.
递归结束条件:当 pos 等于 nums 数组的⻓度时,说明我们已经处理完了所有数字,将当前数组存⼊结果中;
4.
在每个递归状态中,枚举所有下标 i,若这个下标未被标记,并且在它之前的相同元素被标记过,则使⽤ nums 数组中当前下标的元素:
a.
将 check[i] 标记为 true;
b.
将 nums[i] 添加⾄ path 数组末尾;
c.
对第 i+1 个位置进⾏递归;
d.
将 check[i] 重新赋值为 false,并删除 path 末尾元素表⽰回溯;
5.
最后,返回 ret。
例题五
.
解法:
算法思路:
每个位置可选择的字符与其他位置并不冲突,因此不需要标记已经出现的字符,只需要将每个数字对应的字符依次填⼊字符串中进⾏递归,在回溯时撤销填⼊操作即可。
•
在递归之前我们需要定义⼀个字典 hash1,记录 2~9 各⾃对应的字符。
全局变量:
string hash1[10] = { "","","abc","def","ghi","jkl","mno","pqrs","tuv","wxyz" };
vector<string> ret;
string path;
递归函数设计:
void dfs(string digits,int pos)
参数:pos (已经处理的元素个数);
返回值:⽆
函数作⽤:查找所有合理的字⺟组合并存储在答案列表中。
递归函数流程如下:
1.
递归结束条件:当 path 等于 digits 的⻓度时,将 path 加⼊到 ret 中并返回;
2.
取出当前处理的数字 digit,根据 hash1 取出对应的字⺟列表 s;
3.
遍历字⺟列表 s,将当前字⺟加⼊到组合字符串 path 的末尾,然后递归处理下⼀个数字(传
⼊ i + 1,表⽰处理下⼀个数字);
4.
递归处理结束后,将加⼊的字⺟从 path 的末尾删除,表⽰回溯。
5.
最终返回 ret 即可。
例题六
解法:
算法思路:
从左往右进⾏递归,在每个位置判断放置左右括号的可能性,若此时放置左括号合理,则放置左括号继续进⾏递归,右括号同理。
⼀种判断括号是否合法的⽅法:从左往右遍历,左括号的数量始终⼤于等于右括号的数量,并且左括号的总数量与右括号的总数量相等。因此我们在递归时需要进⾏以下判断:
1.
放⼊左括号时需判断此时左括号数量是否⼩于字符串总⻓度的⼀半(若左括号的数量⼤于等于字符串⻓度的⼀半时继续放置左括号,则左括号的总数量⼀定⼤于右括号的总数量);
2.
放⼊右括号时需判断此时右括号数量是否⼩于左括号数量。
全局变量:
vector<string> ret;
string path;
int l, r;
int _n;
递归函数设计:void dfs()
参数:无;
返回值:⽆;
函数作⽤:查找所有合理的括号序列并存储在答案列表中。
递归函数参数设置为当前状态的字符串⻓度以及当前状态的左括号数量,递归流程如下:
1.
递归结束条件:当前r与 n 相等,记录当前状态并返回;
2.
若此时左括号数量⼩于字符串总⻓度的⼀半,则在当前状态的字符串末尾添加左括号并继续递归,递归结束撤销添加操作;
3.
若此时右括号数量⼩于左括号数量(右括号数量可以由当前状态的字符串⻓度减去左括号数量求
得),则在当前状态的字符串末尾添加右括号并递归,递归结束撤销添加操作;
例题七
解法(回溯):
算法思路:
题⽬要求我们从 1 到 n 中选择 k 个数的所有组合,其中不考虑顺序。也就是说,[1,2] 和 [2,1] 等价。我们需要找出所有的组合,但不能重复计算相同元素的不同顺序的组合。对于选择组合,我们需要进⾏
如下流程:
1.
所有元素分别作为⾸位元素进⾏处理;
2.
在之后的位置上同理,选择所有元素分别作为当前位置元素进⾏处理;
3.
为避免计算重复组合,规定选择之后位置的元素时必须⽐前⼀个元素⼤,这样就不会有重复的组合 ([1,2] 和 [2,1] 中 [2,1] 不会出现)。
全局变量:
vector<vector<int>> ret;
vector<int> path;
int _n, _k;
递归函数设计:void dfs(int pos)
参数:pos(当前需要进⾏处理的位置);
返回值:⽆;
函数作⽤:某个元素作为⾸位元素出现时,查找所有可能的组合。
递归流程如下:
a.
结束条件:当前组合中已经有 k 个元素,将当前组合存进⼆维数组并返回。
▪
剪枝:如果当前位置之后的所有元素放⼊组合也不能满⾜组合中存在 k 个元素,直接返回。
b.
从当前位置的下⼀个元素开始遍历到 n,将元素赋值到当前位置,递归下⼀个位置。
例题八
解法(回溯):
算法思路:
对于每个数,可以选择加上或减去它,依次枚举每⼀个数字,在每个数都被选择时检查得到的和是否等于⽬标值。如果等于,则记录结果。
需要注意的是,为了优化时间复杂度,可以提前计算出数组中所有数字的和 sum,以及数组的⻓度 len。这样可以快速判断当前的和减去剩余的所有数是否已经超过了⽬标值 target ,或者当前的和加上剩下的数的和是否⼩于⽬标值 target,如果满⾜条件,则可以直接回溯。
递归流程:
1.
递归结束条件:位置pos 与数组⻓度相等,判断当前状态的 sum 是否与⽬标值相等,若是计数加⼀;
2.
选择当前元素进⾏加操作,递归下⼀个位置,并更新参数 sum;
3.
选择当前元素进⾏减操作,递归下⼀个位置,并更新参数 sum;
例题九
解法:
算法思路:
candidates 的所有元素 互不相同,因此我们在递归状态时只需要对每个元素进⾏如下判断:
1.
跳过,对下⼀个元素进⾏判断;
2.
将其添加⾄当前状态中,我们在选择添加当前元素时,之后仍可以继续选择当前元素(可以重复选择同⼀元素)。
•
因此,我们在选择当前元素并向下传递下标时,应该直接传递当前元素下标。
全局变量:
vector<vector<int>> ret;
vector<int> path;
int _target;
递归函数设计:
void dfs(vector<int>& candidates, int sum,int pos)
参数:sum(当前状态和),pos(当前需要处理的元素下标);
返回值:⽆;
函数作⽤:向下传递两个状态(跳过或者选择当前元素),找出所有组合使得元素和为⽬标值。
递归函数流程如下:
1.
结束条件:
a.
当前需要处理的元素下标越界;
b.
当前状态的元素和已经与⽬标值相同;
2.
跳过当前元素,当前状态不变,对下⼀个元素进⾏处理;
3.
选择将当前元素添加⾄当前状态,并保留状态继续对当前元素进⾏处理,递归结束时撤销添加操
作。
例题十
解法:
算法思路:
只需要对英⽂字⺟进⾏处理,处理每个元素时存在三种情况:
1.
不进⾏处理;
2.
若当前字⺟是英⽂字⺟并且是⼤写,将其修改为⼩写;
3.
若当前字⺟是英⽂字⺟并且是⼩写,将其修改为⼤写。
递归函数设计:void dfs(string& s,int pos)
参数:pos(当前需要处理的位置);
返回值:⽆;
函数作⽤:查找所有可能的字符串集合,并将其记录在答案列表。
从前往后按序进⾏递归,递归流程如下:
1.
递归结束条件:当前需要处理的元素下标越界,表⽰处理完毕,记录当前状态并返回;
2.
对当前元素不进⾏任何处理,直接递归下⼀位元素;
3.
判断当前元素是否为⼩写字⺟,若是,将其修改为⼤写字⺟并递归下⼀个元素,递归结束时撤销修改操作;
4.
判断当前元素是否为⼤写字⺟,若是,将其修改为⼩写字⺟并递归下⼀个元素,递归结束时撤销修改操作;
例题十一
解法:
算法思路:
我们需要在每⼀个位置上考虑所有的可能情况并且不能出现重复。通过深度优先搜索的⽅式,不断地枚举每个数在当前位置的可能性,并回溯到上⼀个状态,直到枚举完所有可能性,得到正确的结果。
我们需要定义⼀个变量 ⽤来记录所有可能的排列数量,⼀个⼀维数组 check 标记元素,然后从第⼀个位置开始进⾏递归;
递归函数设计:void dfs(int pos)
参数:pos(当前需要处理的位置);
返回值:⽆;
函数作⽤:在当前位置填⼊⼀个合理的数字,查找所有满⾜条件的排列。
递归流程如下:
1.
递归结束条件:当 pos 等于 n+1 时,说明已经处理完了所有数字,将当前数组存⼊结果中;
2.
在每个递归状态中,枚举所有下标 x,若这个下标未被标记,并且满⾜题⽬条件之⼀:
a.
将 check[x] 标记为 ture;
b.
对第 pos+1 个位置进⾏递归;
c.
将 check[x] 重新赋值为 false,表⽰回溯;
声明:
本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:
https://www.wpsshop.cn/w/羊村懒王/article/detail/474524
推荐阅读
article
【SQL】
Oracle
和
Mysql
的分页、重复
数据
查询(
limit
、
rownum
、
rowid
)...
上周三面试题有两道涉及
Oracle
的分页查询,没有意外地凉了,现在总结一下。·
Mysql
mysql的分页可以直接使用关...
赞
踩
article
使用
python
编写
android
截屏
脚本
_
android
python
截屏
频率 识图...
测试的过程中经常需要截取屏幕,通常的做法是s_
android
python
截屏
频率 识图
android
python
截屏
...
赞
踩
article
python
常用
库
详解
,
超详细
_
python
各个
库
的
作用...
python
常用
库
详解
,
超详细
,
赶紧一起来学习吧
_
python
各个
库
的
作用
python
各个
库
的
作用 ...
赞
踩
article
达梦
数据
库
表
插入
EXCEL
文件
数据
_
结果
集不可更新
,
请确认查询
列
是否出自同一张
表
,
并且包含值唯一的
列
...
达梦
数据
库
DM管理工具 DMmanger
数据
导入 excel
文件
_
结果
集不可更新
,
请确认查询
列
是否出自同一张
表
,
并且...
赞
踩
article
140种
Python
标准库、
第三方
库和外部
工具
_
python
中
ctypes
是
第三方
模块
嘛...
这些库可用于文件读写、网络抓取和解析、数据连接、数清洗转换、数据计算和统计分析、图像和视频处理、音频处理、数据挖掘/机器...
赞
踩
article
连接
MySQL
错误【
sha2
56
_
passw
or
d
or
caching_
sha2
_passw
or
...
使用django和navicat等连接
MySQL
出现错误:【RuntimeErr
or
: 'cryptography' p...
赞
踩
article
Oracle
分页
查询
慢,sql优化_
oracle
单表
50
万
数据
分页
查询
10000条需要好几秒...
ROWNUM是逻辑地址,表示
查询
耨条记录在整个结果集中的位置,同一条记录
查询
条件不同对应的rownum是不同的二rowi...
赞
踩
article
web
网页
渗透
测试
实验
_
web
系统漏洞
渗透
实验
...
网络
渗透
是攻击者常用的一种攻击手段,也是一种综合的高级攻击技术,同时网络
渗透
中通常被称为”
渗透
测试。其中XSS(Cro...
赞
踩
article
python
国际化
_
python
international...
当 Web服务搭建好以后,可以接收来自全球不同国家用户访问。这样就要求开发人员调整软件,使之能适用于不同的语言,即
国际化
...
赞
踩
article
微信
小
程序
清除
缓存
(ios和安卓
的
解决方法)_加载
小
程序
代码包失败,请尝试
清除
本地
缓存
...
今天在开发
的
时候遇到这样一个问题,服务器上面
的
图片已经上传完毕了,打开相应
的
网址也可以看到对应
的
图片,可是更换到
小
程序
里...
赞
踩
article
一篇文章学会
MySQL
UNION
...
在 SQL 查询中,经常需要从多个数据源获取数据并进行整合分析。
UNION
运算符为此提供了一种强大而灵活的解决方案。本...
赞
踩
article
Vue3.0
官方
文档
...
中文
文档
地址:https://vue3js.cn/docs/zh/_vue3.0官方
文档
vue3.0官方
文档
...
赞
踩
article
【
Git
】常见
代码
冲突
及处理
_
git
冲突
场景
...
在
Git
中,还存在一些其他的
冲突
场景
和解决方案,例如历史版本回退产生
冲突
的情况。原因:Windows操作系统不区分文件名...
赞
踩
article
【
STM3
2
+
HAL
+
Proteus
】
系列学习
教程
2
---
STM3
2
开发
模式选择...
STM3
2
开发
常用的三种模式【
STM3
2
+
HAL
+
Proteus
】
系列学习
教程
2
---
STM3
2
开发
模式选择 ...
赞
踩
article
面试十七、
list
和
deque
...
Deque容器是连续的空间,至少逻辑上看来如此,连续现行空间总是令我们联想到array
和
vector,array无法成长...
赞
踩
article
雷电
模拟器
的
下载
、
安装
与使用_
ubuntu
安装
雷电
模拟器
...
雷电
模拟器
的
下载
雷电
模拟器
的
安装
雷电
模拟器
的使用
雷电
模拟器
的
下载
雷电
模拟器
的
下载
地址为:
雷电
模拟器
雷电
模拟器
的
安装
...
赞
踩
article
【图文教程】
Centos
7
下
安装
Hadoop
_如何在终端
下
载
hadoop
...
比如凯哥的
安装
目录,就是第一步上传到/data后解压的。所以
hadoop
安装
目录就是:/data/
hadoop
-2.7....
赞
踩
article
学习
C++
的常用
网站
_
c++
学习
网站
...
中文:https://zh.cppreference.com/w/英文:http://en.cppreference.c...
赞
踩
article
pm2
管理
Java
_详解使用PM2
管理
nodejs
进程
...
pm2
是一个带有负载均衡功能的Node应用的
进程
管理
器.当你要把你的独立代码利用全部的服务器上的所有CPU,并保证
进程
...
赞
踩
article
SoulPermission
动态
申请
权限
(封装)...
SoulPermission
使用方法适配Android X使用(建议使用):dependencies { impleme...
赞
踩
相关标签
数据库
面试
python
adb
android
测试
脚本
开发语言
excel
Python
oracle
sql
服务器
运维
安全
python 国际化
mysql
vue.js
前端
javascript
git
stm32
proteus
嵌入式硬件