代码随想录算法训练营第二十四天 | 回溯算法理论基础、77. 组合

回溯法是一种通过穷举所有可能的解决方案并递归地尝试来解决问题的搜索算法。它通常用于解决组合、排列、子集和棋盘类问题,如N皇后和数独。由于穷举所有情况,回溯法效率不高。文章介绍了回溯法的基本思想,包括回溯函数的三部曲:定义、终止条件和搜索过程,并通过组合问题的实例展示了回溯法的实现模板。
摘要由CSDN通过智能技术生成

回溯算法理论基础

1、回溯法的定义

回溯法也可以叫做回溯搜索法,它是一种搜索的方式。

回溯是递归的副产品,只要有递归就会有回溯;回溯函数也就是递归函数,指的都是一个函数

2、回溯法的效率

回溯的本质是穷举所有可能出现的情况,因此回溯并不是高效算法

3、回溯法解决的问题

回溯法,一般可以解决如下几种问题:

  • 组合问题:N个数里面按一定规则找出k个数的集合
  • 切割问题:一个字符串按一定规则有几种切割方式
  • 子集问题:一个N个数的集合里有多少符合条件的子集
  • 排列问题:N个数按一定规则全排列,有几种排列方式
  • 棋盘问题:N皇后,解数独

组合是不强调元素顺序的,排列是强调元素顺序

4、如何理解回溯法

回溯法所要解决的问题可以抽象为树形结构;因为回溯法解决的都是在集合中递归查找子集,集合的大小构成了树的宽度,递归的深度构成了树的深度

5、回溯法模板

回溯三部曲:

1、回溯函数模板返回值以及参数--函数起名字为backtracking;回溯算法中函数返回值一般为void;回溯算法需要的参数可不像二叉树递归的时候那么容易一次性确定下来,所以一般是先写逻辑,然后需要什么参数,就填什么参数

void backtracking(参数)

2、回溯函数终止条件--什么时候达到了终止条件,树中就可以看出,一般来说搜到叶子节点了,也就找到了满足条件的一条答案,把这个答案存放起来,并结束本层递归

if (终止条件) {
    存放结果;
    return;
}

3、回溯搜索的遍历过程--回溯法一般是在集合中递归搜索,集合的大小构成了树的宽度,递归的深度构成的树的深度

for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {
    处理节点;
    backtracking(路径,选择列表); // 递归
    回溯,撤销处理结果
}

77. 组合

采用一般暴力解法就是普通的嵌套循环,但随着k的增大,循环嵌套只会增加时间复杂度

因此本题采用回溯法,n是树的宽度,k是树的宽度

每次从集合中选取元素,可选择的范围随着选择的进行而收缩,调整可选择的范围

回溯三部曲

1、递归函数的返回值以及参数----函数里参数太多影响可读性,因此定义两个全局变量,一个用来存放符合条件单一结果,一个用来存放符合条件结果的集合;函数里一定有两个参数,既然是集合n里面取k个数,那么n和k是两个int型的参数;还需要一个参数,为int型变量startIndex,这个参数用来记录本层递归的中,集合从哪里开始遍历(集合就是[1,...,n] )

vector<vector<int>> result; // 存放符合条件结果的集合
vector<int> path; // 用来存放符合条件单一结果
void backtracking(int n, int k, int startIndex) 

2、回溯函数终止条件----path这个数组的大小如果达到k,说明我们找到了一个子集大小为k的组合了,在图中path存的就是根节点到叶子节点的路径

if (path.size() == k) {
    result.push_back(path);
    return;
}

3、单层搜索的过程----回溯法的搜索过程就是一个树型结构的遍历过程,for循环用来横向遍历,递归的过程是纵向遍历

for (int i = startIndex; i <= n; i++) { // 控制树的横向遍历
    path.push_back(i); // 处理节点 
    backtracking(n, k, i + 1); // 递归:控制树的纵向遍历,注意下一层搜索要从i+1开始
    path.pop_back(); // 回溯,撤销处理的节点
}

本题代码如下:

class Solution {
public:

    vector<vector<int>> result; //用于存放各种结果集
    vector<int> path; //用于记录单次集合

    void backtracking(int n, int k, int startInde) {
        if (path.size() == k) {
            result.push_back(path);
            return ;
        }

        for (int i = startInde; i <= n; i++) {
            path.push_back(i);
            backtracking(n, k, i + 1);
            path.pop_back();
        }
    }

    vector<vector<int>> combine(int n, int k) {
        result.clear();
        path.clear();
        backtracking(n, k, 1);
        return result;
    }
};

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
代码随想录算法训练营是一个优质的学习和讨论平台,提供了丰富的算法训练内容和讨论交流机会。在训练营中,学员们可以通过观看视频讲解来学习算法知识,并根据讲解内容进行刷题练习。此外,训练营还提供了刷题建议,例如先看视频、了解自己所使用的编程语言、使用日志等方法来提高刷题效果和语言掌握程度。 训练营中的讨论内容非常丰富,涵盖了各种算法知识点和解题方法。例如,在第14训练营中,讲解了二叉树的理论基础、递归遍历、迭代遍历和统一遍历的内容。此外,在讨论中还分享了相关的博客文章和配图,帮助学员更好地理解和掌握二叉树的遍历方法。 训练营还提供了每日的讨论知识点,例如在第15的讨论中,介绍了层序遍历的方法和使用队列来模拟一层一层遍历的效果。在第16的讨论中,重点讨论了如何进行调试(debug)的方法,认为掌握调试技巧可以帮助学员更好地解决问题和写出正确的算法代码。 总之,代码随想录算法训练营是一个提供优质学习和讨论环境的平台,可以帮助学员系统地学习算法知识,并提供了丰富的讨论内容和刷题建议来提高算法编程能力。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [代码随想录算法训练营每日精华](https://blog.csdn.net/weixin_38556197/article/details/128462133)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值