力扣算法刷题 Day22

回溯算法

回溯算法是用于搜索的一种算法,伴随着递归。所有需要回溯法解决的问题数据都可以抽象为树的数据结构。

既然需要在集合中递归查找子集,那么集合的大小构成了树的宽度,递归的深度决定了树的深度。

算法效率不高,因为本质是暴力穷举,顶多剪枝优化。

常用于解决的问题:

算法模板

cpp 复制代码
void backtracking(参数) {
    if (终止条件) {
        存放结果;
        return;
    }

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

77 组合

题目链接

77 组合

思路

用树的方法去思考:从4个数里取出两个数。先取1个,4个分支...

辅助函数返回值为空,参数为开始下标,n,k。递归出口为,当前结果数量满足2,就返回。

单层递归逻辑为:当前层里所有元素先加入结果集,处理后再取出实现回溯。

文章详解

文章详解

cpp 复制代码
class Solution {
public:
    vector<vector<int>> results;
    vector<int> res;
    void tracebacking(int startIndex,int n,int k)
    {
        if(res.size() == k)
        {
            results.push_back(res);
            return;
        }

        for(int i = startIndex; i <= n; i++)
        {
            res.push_back(i);
            tracebacking(i+1,n,k);
            res.pop_back();
        }
    }
    vector<vector<int>> combine(int n, int k) {
        tracebacking(1,n,k);
        return results;
    }
};

216 组合总和III

题目链接

点击此处

思路

和上一道的思路比较接近。回溯算法返回值void,参数:开始数,k,n。结束出口:当前group的剩余总和小于0(采用总和扣除的方式),或group的元素个数大于k,直接返回;当前group的剩余总和为0,如果group的元素个数为k,则加入结果集,并且返回;否则直接返回。单层递归逻辑:遍历当前层元素(注意从start到9),依次入集出集。

文章详解

点击此处

cpp 复制代码
class Solution {
public:
    vector<vector<int>> result;
    vector<int> group;
    void tracebacking(int start, int k, int n)
    {
        if(n < 0 || group.size() > k)
        {
            return;
        }
        if(n == 0)
        {
            if(group.size() == k)
            {
                result.push_back(group);
            }
            return;
        }

        for(int i = start; i <= 9; i++)
        {
            group.push_back(i);
            tracebacking(i+1,k,n-i);
            group.pop_back();
        }
    }
    vector<vector<int>> combinationSum3(int k, int n) {
        tracebacking(1,k,n);
        return result;
    }
};

17 电话号码的字母组合

题目链接

点击此处

思路

我们可以先建立一个string map数组,下标对应记录2-9的映射字符串集合,用于从中取字符。中间处理注意将字符串数字转为int类型来对应下标。

文章详解

点击此处

cpp 复制代码
class Solution {
private:
    const string letterMap[10] = {
        "", // 0
        "", // 1
        "abc", // 2
        "def", // 3
        "ghi", // 4
        "jkl", // 5
        "mno", // 6
        "pqrs", // 7
        "tuv", // 8
        "wxyz", // 9
    };
public:
    vector<string> results;
    string s;
    void tracebacking(const string &digits,int index)
    {
        if(index == digits.length())
        {
            results.push_back(s);
            return;
        }
        int digit = digits[index] - '0';
        string letter = letterMap[digit];
        for(int i = 0;i< letter.length();i++)
        {
            s.push_back(letter[i]);
            tracebacking(digits,index+1);
            s.pop_back();
        }
    }
    vector<string> letterCombinations(string digits) {
        if(digits.length() == 0)
        {
            return results;
        }
        tracebacking(digits,0);
        return results;
    }
};
相关推荐
贾斯汀玛尔斯21 小时前
每天学一个算法--LSM-Tree(Log-Structured Merge Tree)
java·算法·lsm-tree
浅念-1 天前
刷穿LeetCode:BFS 解决 Flood Fill 算法
数据结构·c++·算法·leetcode·职场和发展·bfs·宽度优先
做cv的小昊1 天前
【TJU】研究生应用统计学课程笔记(8)——第四章 线性模型(4.1 一元线性回归分析)
笔记·线性代数·算法·数学建模·回归·线性回归·概率论
贾斯汀玛尔斯1 天前
每天学一个算法--倒排索引(Inverted Index)
算法·inverted-index
小e说说1 天前
打破偏科困境:这些学习软件助孩子重燃学习热情
算法
我命由我123451 天前
程序员的心理学学习笔记 - 空杯心态
经验分享·笔记·学习·职场和发展·求职招聘·职场发展·学习方法
月昤昽1 天前
autoCAD二次开发 4.正多边形与collection区分
算法·c#·二次开发·autocad二次开发
休息一下接着来1 天前
C++ 固定容量环形队列实现
c++·算法
im_AMBER1 天前
手撕hot100之矩阵!看完这篇就AC~
javascript·数据结构·线性代数·算法·leetcode·矩阵