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

目录

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

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

​编辑回溯三部曲

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皇后,解数独等等

相关推荐
hunterkkk(c++)1 小时前
学习dijkstra算法(c++)
c++·学习·算法
Chensay.1 小时前
链表的基本操作(C语言版)
数据结构·链表
lightqjx1 小时前
【算法】数据结构_单调队列
数据结构·算法·单调队列
c++之路1 小时前
Linux 下 C++ 开发环境搭建
linux·运维·c++
小四季豆1 小时前
《数据结构与算法》-顺序表:算法落地的第一个线性结构
c语言·数据结构·算法
8Qi82 小时前
LeetCode 96:不同的二叉搜索树(Unique Binary Search Trees)—— 题解 ✅
算法·leetcode·职场和发展·动态规划
189228048612 小时前
NV041固态MT29F16T08GSLCEM9-QBES:C
人工智能·算法·microsoft·缓存·性能优化
罗超驿2 小时前
15.LeetCode 30. 串联所有单词的子串(Java):滑动窗口+哈希表详解
算法·leetcode
Marianne Qiqi2 小时前
非hot100的力扣算法题
数据结构·算法·leetcode