【高频考点】回溯(暴力搜索)

目录

回溯的目标:是要返回所有符合条件的结果的集合(暴力搜索)

回溯的遍历(类似树的遍历):

​编辑回溯三部曲

1递归函数的返回值以及参数

2回溯函数终止条件

3单层搜索的过程

经典问题

组合问题:N个数里面按一定规则找出k个数的集合

[切割问题:一个字符串按一定规则有几种切割方式 abcdef](#切割问题:一个字符串按一定规则有几种切割方式 abcdef)

子集问题:一个N个数的集合里有多少符合条件的子集

排列问题:N个数按一定规则全排列,有几种排列方式

棋盘问题:N皇后,解数独等等


回溯的目标:是要返回所有符合条件的结果的集合(暴力搜索)

回溯的遍历(类似树的遍历):

for循环-横向遍历-集合大小(树的宽度)

递归-纵向遍历-树的深度

回溯三部曲

1递归函数的返回值以及参数

2回溯函数终止条件

3单层搜索的过程

经典模板:

cpp 复制代码
void backtracking(参数) {
    if (终止条件) {
        存放结果;
        return;
    }
​
    for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {
        处理节点;   //处理
        backtracking(路径,选择列表); // 递归
        回溯,撤销处理结果   //还原现场
    }
}

解题技巧:返回值一般是void;先写逻辑再填参数

经典问题

组合问题:N个数里面按一定规则找出k个数的集合

1)写出集合数组 是1,2,3,...,n;画出树结构

2)写回溯函数:①终止条件:长度为k时,加到结果里并返回

②遍历:for范围是集合中元素的个数;

递归的索引是从当前的下一个元素开始(因组合无序)

3)组合数是无序的(用过的不能再用):因此for遍历起始值是当前元素的idx,递归的索引是从当前的下一个元素开始

*如果是多个集合取组合,各个集合之间相互不影响,那么就不用idx

*如果元素是可重复选取的,递归的i的idx不需要+1

*如果集合中元素有重复,但只能用集合中的,就需要用used标记/idx去重;还要先排序sort(candidates.begin(), candidates.end()); 操作的时候加判断是否与上个数字重复

4)定义全局变量:结果数组和中间结果数组

cpp 复制代码
class Solution {
    vector<vector<int>> result;// 存放符合条件结果的集合
    vector<int> temp;// 用来存放符合条件结果
public:
    void backtracking(vector<int> nums, int n, int k,int idx){
        if(temp.size()==k){
            result.push_back(temp);
            return;
        }
        for(int i=idx;i<n;i++){//剪枝优化 i<n-(k-temp.size())+1
            temp.push_back(nums[i]);
            backtracking(nums,n,k,i+1);
            temp.pop_back();
            }  
        }
    vector<vector<int>> combine(int n, int k) {
        vector<int> nums;
        for(int i=0;i<n;i++){
            nums.push_back(i+1);
        }
        backtracking(nums,n,k,0);
        return result;
    }
};

优化:如果for循环选择的起始位置之后的元素个数已经不足我们需要的元素个数了,那么就没有必要搜索了
*

切割问题:一个字符串按一定规则有几种切割方式 abcdef

-选取一个a之后,在bcdef中再去选取第2个,选取b之后在cdef中再选取第3个。

-切割一个a之后,在bcdef中再去切割第2段,切割b之后在cdef中再切割第3段。

cpp 复制代码
// 获取[startIndex,i]在s中的子串
string str = s.substr(startIndex, i - startIndex + 1);

子集问题:一个N个数的集合里有多少符合条件的子集

相当于没有终止条件的组合,是统计所有节点(组合是统计所有叶子节点)
*

排列问题:N个数按一定规则全排列,有几种排列方式

每层都是从0开始搜索而不是startIndex

需要used数组记录temp里都放了哪些元素了
*

棋盘问题:N皇后,解数独等等

相关推荐
千纸鹤安安4 小时前
千问Qwen-AgentWorld来了:一个语言模型搞定七大Agent场景,GPT-5.4都输了
算法
七牛开发者6 小时前
MCP 到底是什么?为什么 Agent 都想接上它
算法·aigc·agent
北域码匠10 小时前
冒泡排序太慢?鸡尾酒排序双向优化,原生 C# 零第三方库完整代码
数据结构·排序算法·泛型·c# 算法·鸡尾酒排序·原生 c# 开发·冒泡排序优化·嵌入式算法
卷无止境11 小时前
C++ 的Eigen 库全解析
c++
卷无止境11 小时前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端
郝学胜_神的一滴12 小时前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake
kisshyshy13 小时前
从递归到迭代,一文吃透二叉树的核心知识与 JavaScript 实现
javascript·算法·代码规范
To_OC1 天前
LC 49 字母异位词分组:想到哈希表很简单,选对 key 才是精髓
javascript·算法·leetcode