Backtracking 回溯算法
引入:
当暴力穷举很好解决的话,回溯就是就是一个实现方法。
原理:
① 什么是回溯法
回溯法也可以叫做回溯搜索法,它是一种搜索的方式。
在二叉树系列中,我们已经不止一次,提到了回溯,例如递归,回溯——2022年8月9日21:41:20 - 二叉树的所有路径 - 力扣(LeetCode)。
回溯是递归的副产品,只要有递归就会有回溯。
所以以下讲解中,回溯函数也就是递归函数,指的都是一个函数。
② 回溯法的效率
回溯法的性能如何呢,这里要和大家说清楚了,虽然回溯法很难,很不好理解,但是回溯法并不是什么高效的算法。因为回溯的本质是穷举,穷举所有可能,然后选出我们想要的答案,如果想让回溯法高效一些,可以加一些剪枝的操作,但也改不了回溯法就是穷举的本质。
那么既然回溯法并不高效为什么还要用它呢?
因为没得选,一些问题能暴力搜出来就不错了,撑死了再剪枝一下,还没有更高效的解法。
此时大家应该好奇了,都什么问题,这么牛逼,只能暴力搜索。
③ 回溯法解决的问题
回溯法,一般可以解决如下几种问题:
- 组合问题:N个数里面按一定规则找出k个数的集合
- 切割问题:一个字符串按一定规则有几种切割方式
- 子集问题:一个N个数的集合里有多少符合条件的子集
- 排列问题:N个数按一定规则全排列,有几种排列方式
- 棋盘问题:N皇后,解数独等等
相信大家看着这些之后会发现,每个问题,都不简单!
什么是组合,什么是排列?
组合是不强调元素顺序的,排列是强调元素顺序。
例如:{1, 2} 和 {2, 1} 在组合上,就是一个集合,因为不强调顺序,而要是排列的话,{1, 2} 和 {2, 1} 就是两个集合了。
记住组合无序,排列有序,就可以了。
④ 如何理解回溯法
回溯法解决的问题都可以抽象为树形结构,是的,我指的是所有回溯法的问题都可以抽象为树形结构!
因为回溯法解决的都是在集合中递归查找子集,集合的大小就构成了树的宽度,递归的深度,都构成的树的深度。
递归就要有终止条件,所以必然是一棵高度有限的树(N叉树)。
这块可能初学者还不太理解,后面的回溯算法解决的所有题目中,我都会强调这一点并画图举相应的例子,现在有一个印象就行。
模板:
在讲BinomialTree 二叉树-CSDN博客中我们说了递归三部曲,这里我再给大家列出回溯三部曲。
① 回溯函数模板返回值以及参数
在回溯算法中,我的习惯是函数起名字为backtracking,这个起名大家随意。
回溯算法中函数返回值一般为void。
再来看一下参数,因为回溯算法需要的参数可不像二叉树递归的时候那么容易一次性确定下来,所以一般是先写逻辑,然后需要什么参数,就填什么参数。
但后面的回溯题目的讲解中,为了方便大家理解,我在一开始就帮大家把参数确定下来。
回溯函数伪代码如下:
void backtracking(参数)
② 回溯函数终止条件
既然是树形结构,那么我们在讲解BinomialTree 二叉树-CSDN博客的时候,就知道遍历树形结构一定要有终止条件。
所以回溯也有要终止条件。
什么时候达到了终止条件,树中就可以看出,一般来说搜到叶子节点了,也就找到了满足条件的一条答案,把这个答案存放起来,并结束本层递归。
所以回溯函数终止条件伪代码如下:
if (终止条件) {存放结果;return;
}
③ 回溯搜索的遍历过程
在上面我们提到了,回溯法一般是在集合中递归搜索,集合的大小构成了树的宽度,递归的深度构成的树的深度。
如图:
注意图中,我特意举例集合大小和孩子的数量是相等的!
回溯函数遍历过程伪代码如下:
for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {处理节点;backtracking(路径,选择列表); // 递归回溯,撤销处理结果
}
for循环就是遍历集合区间,可以理解一个节点有多少个孩子,这个for循环就执行多少次。
backtracking这里自己调用自己,实现递归。
大家可以从图中看出for循环可以理解是横向遍历,backtracking(递归)就是纵向遍历,这样就把这棵树全遍历完了,一般来说,搜索叶子节点就是找的其中一个结果了。
分析完过程,回溯算法模板框架如下:
void backtracking(参数) {if (终止条件) {存放结果;return;}for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {处理节点;backtracking(路径,选择列表); // 递归回溯,撤销处理结果}
}
这份模板很重要,后面做回溯法的题目都靠它了!
题目实战:
① 组合问题:(单体中元素个数是一致的)
回溯+递归+剪枝优化——2022年8月10日22:48:38 - 组合 - 力扣(LeetCode)
递归,回溯,剪枝——2022年8月11日01:03:04 - 组合总和 III - 力扣(LeetCode)
回溯,组合——2022年8月11日13:15:09 - 电话号码的字母组合 - 力扣(LeetCode)
回溯+if剪枝+for循环剪枝——2022年8月11日13:50:08 - 组合总和 - 力扣(LeetCode)
回溯,剪枝,同层元素去重——2022年8月11日14:15:00 - 组合总和 II - 力扣(LeetCode)
② 切割问题:(startIndex的作用)
例如对于字符串abcdef:
- 组合问题:选取一个a之后,在bcdef中再去选取第二个,选取b之后在cdef中在选组第三个…。
- 切割问题:切割一个a之后,在bcdef中再去切割第二段,切割b之后在cdef中在切割第三段…。
代码中startIndex一般就是切割线。
回溯,递归,回文串判断——2022年8月11日15:01:31 - 分割回文串 - 力扣(LeetCode)
回溯,分割字符串,剪枝——2022年8月11日17:39:11 - 复原 IP 地址 - 力扣(LeetCode)
③ 子集问题:(全遍历)
如果把 子集问题、组合问题、分割问题都抽象为一棵树的话,那么组合问题和分割问题都是收集树的叶子节点,而子集问题是找树的所有节点!
求取子集问题,不需要任何剪枝!因为子集就是要遍历整棵树。
Arrays.sort(nums);//回溯中间,一般用双指针来去重,就需要先对数组进行排序操作。
特殊的去重逻辑。根据题意咯。
回溯,子集问题——2022年8月11日20:26:04 - 子集 - 力扣(LeetCode)
回溯,层级去重——2022年8月11日20:37:14 - 子集 II - 力扣(LeetCode)
回溯,子集问题,同层去重——2022年8月11日21:18:35 - 递增子序列 - 力扣(LeetCode)
④ 排列问题:(startIndex = 0)
//而used数组,其实就是记录此时path里都有哪些元素使用了,一个排列里一个元素只能使用一次。
//是纵向的,所以used数组需要回溯。全排列一般就是从0开始了。
回溯,全排列,哈希——2022年8月11日21:37:55 - 全排列 - 力扣(LeetCode)
回溯,双重去重,双指针——2022年8月11日22:09:52 - 全排列 II - 力扣(LeetCode)
⑤ 棋盘问题:
N皇后:
回溯,N皇后棋盘,数组的简单操作——2022年8月12日15:09:37 - N 皇后 - 力扣(LeetCode)
数独:
回溯,递归,DFS,二维递归——2022年8月12日15:58:20 - 解数独 - 力扣(LeetCode)
⑥ 其他:
DFS,回溯,数据结构,Map——2022年8月12日14:32:01 - 重新安排行程 - 力扣(LeetCode)
总结:
本篇我们讲解了,什么是回溯算法,知道了回溯和递归是相辅相成的。
接着提到了回溯法的效率,回溯法其实就是暴力查找,并不是什么高效的算法。
然后列出了回溯法可以解决几类问题,可以看出每一类问题都不简单。
最后我们讲到回溯法解决的问题都可以抽象为树形结构(N叉树),并给出了回溯法的模板。
参考链接:
带你学透回溯算法(理论篇) (opens new window)一起学习!
Backtracking 回溯算法相关推荐
- python数独伪代码回溯法_数独 #回溯算法 #CTF
1. intro:巅峰极客的一道逆向 刷巅峰极客2020里的rev题fu!kpy,复杂得不行但是看到if d[1][0] != '8' or d[1][7] != '2'和if check(h1) ! ...
- 回溯算法(Backtracking Algorithm)之八皇后问题
文章目录 1. 回溯算法思想 2. 算法应用 2.1 八皇后问题 1. 回溯算法思想 前面讲过贪心算法并不能保证得到最优解,那怎么得到最优解呢? 回溯思想,有点类似枚举搜索.枚举所有的解,找到满足期望 ...
- 4.6 Heuristics for Backtracking Algorithms回溯算法的启发式
当使用回溯搜索解决CSP时,必须对要分支或实例化的变量以及要给该变量的值做出一系列决策.这些决策称为变量和值排序.已有研究表明,对于许多问题,变量的选择和值的排序对于有效解决问题是至关重要的(如[5, ...
- 用栈、回溯算法设计迷宫程序
目录 1.走迷宫与回溯算法 2.迷宫设计栈扮演的角色 3.Python实现走迷宫 栈的应用有许多,本篇博文着重将栈与回溯(Backtracking)算法结合,设计走迷宫程序.其实回溯算法也是人工智能的 ...
- 数字拆分问题算法回溯_回溯算法:求子集问题!
给「代码随想录」一个星标吧! ❝ 认识本质之后,这就是一道模板题 通知:我将公众号文章和学习相关的资料整理到了Github :https://github.com/youngyangyang04/le ...
- 回溯 皇后 算法笔记_回溯算法:N皇后问题
给「代码随想录」一个星标吧! ❝ 通知:我将公众号文章和学习相关的资料整理到了Github :https://github.com/youngyangyang04/leetcode-master,方便 ...
- python回溯算法_什么是回溯法,Python解法交流?
只有去多做题,才能慢慢掌握.力扣leetcode-cn.com LeetCode 上的解释 回溯算法实际上一个类似枚举的搜索尝试过程,主要是在搜索尝试过程中寻找问题的解,当发现已不满足求解条件时,就 ...
- 求n个数中第k大的数_互联网高频面试题目:「回溯算法」求组合总和
我将算法学习相关的资料已经整理到了Github :https://github.com/youngyangyang04/leetcode-master,里面还有leetcode刷题攻略.各个类型经典题 ...
- 113. 路径总和 (剑指 Offer 34. 二叉树中和为某一值的路径)(回溯算法)
给你二叉树的根节点 root 和一个整数目标和 targetSum ,找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径. 叶子节点 是指没有子节点的节点. 示例 1: 输入:root = [ ...
最新文章
- system volume information怎么删除_文件系统怎么让Linux内核认识自己
- Android 5.x 权限问题解决方法
- java 对外提供接口_Java服务器对外提供接口以及Android端向服务器请求数据
- python: Failed calling sys.__interactivehook__ (Windows)
- 雅诗兰黛天猫超级品牌日:未央唇膏、红装小棕瓶“当红不让”
- 使用ArcGIS Server发布我们的数据
- [问题解决]同时显示多个Notification时PendingIntent的Intent被覆盖?
- 【英语学习】【科学】【Glencoe Science】【B】From Bacteria to Plants 目录及术语表
- Linux命令解释之mkfs.ext3
- linux 脚本map,shell中map的用法
- 2021-0(C++)输入一个字符串,判断其是否是回文字符串(回文字符串就是正序与反序是相同的字符串)5-27
- java调用百度语音接口
- STM32自学笔记-8-红外遥控
- 人力资源管理-各类激励理论
- Python爬虫——爬取豆瓣VIP书籍信息并存入数据库
- VB.NET创建/修复/压缩/备份/恢复ACCESS数据库
- Android补间动画使用
- G2、D3 绘制维恩图
- 【EtherCAT理论篇】一、EtherCAT现场总线概述
- 攻防世界 web高手进阶区 8分题 Web_python_block_chain