算法训练营第二十八天 | LeetCode 77 组合(剪枝优化)、LeetCode 216 组合总和III、LeetCode 17 电话号码的字母组合

LeetCode 77 组合(剪枝优化)

当我们到达某一层,后面的结点数已经不能满足条件时。可以进行剪枝操作。

代码如下:

cpp 复制代码
class Solution {
private:
    vector<int> path;
    vector<vector<int>> res;
    void backtracking(int n, int index, int k) {
        if (path.size() == k) {
            res.push_back(path);
            return;
        }
        if (index == n + 1) return;
        for (int i = index; i <= n; i++) {
            if (n - i + 1 < k - path.size()) return;
            path.push_back(i);
            backtracking(n, i + 1, k);
            path.pop_back();
        }
    }
public:
    vector<vector<int>> combine(int n, int k) {
        backtracking(n, 1, k);
        return res;
    }
};

主要在for循环中加了一个判断条件。

LeetCode 216 组合总和III

其实也很简单,回溯问题主要是要掌握递归模板,之后的过程就是在凑参数、返回条件和递归逻辑了。具体可以去看看代码随想录里面讲解,会很懂很多。有很多图片,比如下面这个

代码如下:

cpp 复制代码
class Solution {
private:
    vector<int> path;
    vector<vector<int>> res;
    void backtracking(int k, int num, int index) {
        if (path.size() == k) {
            if (num == 0) {
                res.push_back(path);
            }
            return;
        }
        for (int i = index; i <= 9; i++) {
            if (9 - i + 1 < k - path.size()) return;
            path.push_back(i);
            backtracking(k, num - i, i + 1);
            path.pop_back();
        }
    }
public:
    vector<vector<int>> combinationSum3(int k, int n) {
        backtracking(k, n, 1);
        return res;
    }
};

LeetCode 17 电话号码的字母组合

这题本身其实不难,不过有点绕。要用一个字符串类型向量记录每个电话号码数字对应的几个字符组成的字符串,在回溯的时候对应查找,作为查找表使用。

转化时用到stoi和atoi,都报错,最后用字符串减法,直接每个字符-'0'解决了问题。查资料可以发现,两者都是适配于字符串类型而非单个字符的,差别在于一个是c语言函数,一个是C++下的string类里的函数。

代码如下:

cpp 复制代码
class Solution {
private:
    string path;
    vector<string> res;
    void backtracking(string digits, int index, vector<string>& s) {
        if (path.size() == digits.size()) {
            res.push_back(path);
            return;
        }
        int i_index = digits[index] - '0';
        for (int i = 0; i < s[i_index].length(); i++) {
            string temp = path;
            path += s[i_index][i];
            backtracking(digits, index + 1, s);
            path = temp;
        }
    }
public:
    vector<string> letterCombinations(string digits) {
        if (digits.size() == 0) return res;
        vector<string> s(10, "");
        s[2] = "abc"; s[3] = "def"; s[4] = "ghi"; s[5] = "jkl"; s[6] = "mno";
        s[7] = "pqrs"; s[8] = "tuv"; s[9] = "wxyz";
        backtracking(digits, 0, s);
        return res;
    }
};
相关推荐
释怀°Believe3 小时前
Daily算法刷题【面试经典150题-7️⃣位运算/数学/】
算法·面试·职场和发展
2401_876221343 小时前
因数个数、因数和、因数积
c++·算法
云里雾里!3 小时前
LeetCode 744. 寻找比目标字母大的最小字母 | 从低效到最优的二分解法优化
算法·leetcode
一条大祥脚3 小时前
26.1.3 快速幂+容斥 树上dp+快速幂 带前缀和的快速幂 正序转倒序 子序列自动机 线段树维护滑窗
数据结构·算法
二狗哈3 小时前
czsc入门5: Tick RawBar(原始k线) NewBar (新K线)
算法·czsc
꧁Q༒ོγ꧂3 小时前
算法详解(四)--排序与离散化
数据结构·算法·排序算法
Tisfy4 小时前
LeetCode 0865.具有所有最深节点的最小子树:深度优先搜索(一次DFS + Python5行)
算法·leetcode·深度优先·dfs·题解
Q741_1474 小时前
C++ 队列 宽度优先搜索 BFS 力扣 429. N 叉树的层序遍历 C++ 每日一题
c++·算法·leetcode·bfs·宽度优先
Yzzz-F4 小时前
P4145 上帝造题的七分钟 2 / 花神游历各国[线段树 区间开方(剪枝) + 区间求和]
算法·机器学习·剪枝
Zzz不能停4 小时前
堆排序算法及大小堆区别
数据结构·算法