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

回溯算法理论基础

文章讲解:代码随想录#回溯算法理论基础
视频讲解:带你学透回溯算法(理论篇)| 回溯法精讲!

什么是回溯法

回溯法也叫做回溯搜索法,是一种搜索的方式。
回溯是递归的副产品,只要有递归就会有回溯。
回溯法的效率并不高,它的本质就是穷举法,有时候也会有剪枝的操作。

有些问题只有通过暴力穷举才能解决,比如可以解决以下问题:
在这里插入图片描述

回溯法的理解

回溯法解决的问题都可以抽象成一个树形结构。
回溯法解决的都是在集合中递归查找子集,集合的大小就构成了树的宽度,递归的深度就构成了树深度。
由于递归有终止条件,所以它是一棵高度有限的树。

回溯法模板

递归有三部曲,同理回溯也有三部曲。

  • 回溯函数体的返回值以及参数
    回溯算法中函数返回值一般为void。
    参数不能提前确定的,需要在根据处理逻辑来确定参数。
    所以回溯函数代码如下
void backtracking(参数)
  • 回溯函数终止条件
    一般情况下搜到叶子节点就找到了满足条件的一种解决方法,需要将这个方法保存起来,同时要结束本层递归。
if (终止条件) {
   
    存放结果;
    return;
}
  • 回溯搜索的遍历过程
    回溯一般都是在集合中递归搜索 ,集合的大小构成了树的宽度,递归的深度构成了树的深度。
for(选择:本层集合中元素(树中节点孩子的数量就是集合的大小)){
   
	处理节点;
	backtracking(路径,选择列表); // 继续递归
	回溯处理;
}

for循环就是遍历集合区间,可以理解一个节点有多少个孩子,这个for循环就会执行多少次。
其实,for循环就是横向遍历,递归就是纵向遍历。

回溯算法模板框架如下:

void backtracking(参数) {
   
    if (终止条件) {
   
        存放结果;
        return;
    }

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

LeetCode 77.组合

题目链接:77.组合
文章讲解:代码随想录#77.组合
视频讲解:带你学透回溯算法-组合问题(对应力扣题目:77.组合)| 回溯法精讲!

题目描述

给定两个整数 n 和 k,返回范围 [1, n] 中所有可能的 k 个数的组合。

你可以按 任何顺序 返回答案。

示例1

输入:n = 4, k = 2
输出:
[
[2,4],
[3,4],
[2,3],
[1,2],
[1,3],
[1,4],
]

示例2

输入:n = 1, k = 1
输出:[[1]]

提示

  • 1 <= n <= 20
  • 1 <= k <= n

思路

这是一道经典的回溯题,求的是组合,并非排列。
对于组合,【1,2】和【2,1】是一回事,对于排列【1,2】和【2,1】不相同。
组合是不强调元素顺序的,排列是强调元素顺序。
所以,这道题中某个元素进行过组合后,就需要不能再重复计算了。

那如何使用回溯算法呢?
上面说过回溯的问题都可以抽象成树形结构,盗图说明一下。
在这里插入图片描述
n相当于树的宽度,k相当于树的深度,每次搜索到叶子节点就表示找到了一个结果。

参考代码

typedef struct {
   
    int index;
    int num[100];
}Result;

Result result = {
   0};
int **res = NULL;
int cnt = 0;

void backtracking(int n, int k, int idx)
{
   
    if (result.index == k) {
    // 终止条件,当result中已经放入了k个元素时
        res[cnt] = (int*)malloc(k * sizeof(int));
        for(int i = 0; i < k; i++) {
   
            res[cnt][i] = result.num[i];
        }
        cnt++;
        return;
    }

    for (int i = idx; i <= n; i++) {
    // 相当于树的横向遍历
        result.num[result.index++] = i; // 处理节点
        backtracking(n, k, i + 1); // 递归遍历下一层
        result.index--; // 回溯
        result.num[result.index] = 0;

    }
}

int** combine(int n, int k, int* returnSize, int** returnColumnSizes) {
   
    res = (int**)malloc(10000 * sizeof(int*));

    backtracking(n, k, 1);

    *returnSize = cnt;
    *returnColumnSizes = (int*)malloc(sizeof(int) * cnt); // 需要给returnColumnSizes分配内存
    for (int i = 0; i < cnt; i++) {
   
        (*returnColumnSizes)[i] = k;
    }
    return res;
}

总结

  1. 代码编译报这个错误,网上查到说明变量没有有效初始化,排查半天还是没有发现问题出在哪儿。
    在这里插入图片描述

优化版本

待补充

最近更新

  1. TCP协议是安全的吗?

    2024-02-22 01:54:01       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-02-22 01:54:01       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-02-22 01:54:01       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-02-22 01:54:01       20 阅读

热门阅读

  1. 编程笔记 Golang基础 010 常量和变量

    2024-02-22 01:54:01       30 阅读
  2. YOLOV8改进系列指南

    2024-02-22 01:54:01       33 阅读
  3. C++程序设计学习笔记(一)

    2024-02-22 01:54:01       24 阅读
  4. 【开源软件的影响力有多大?】

    2024-02-22 01:54:01       30 阅读
  5. 让图片说话SadTalker

    2024-02-22 01:54:01       33 阅读
  6. 嵌入式学习day22 Linux

    2024-02-22 01:54:01       28 阅读
  7. Linux--shell编程中的for循环

    2024-02-22 01:54:01       35 阅读
  8. SQL 和 NoSQL 有什么区别?

    2024-02-22 01:54:01       24 阅读
  9. 【菜鸡常见网络问题汇总】之:ARP详解

    2024-02-22 01:54:01       37 阅读
  10. 运动重定向学习笔记

    2024-02-22 01:54:01       29 阅读
  11. 数据安全:证书和密钥对概念详解

    2024-02-22 01:54:01       31 阅读