力扣爆刷第123天之回溯五连刷

力扣爆刷第123天之回溯五连刷

文章目录

      • 力扣爆刷第123天之回溯五连刷
      • [一、77. 组合](#一、77. 组合)
      • [二、216. 组合总和 III](#二、216. 组合总和 III)
      • [三、17. 电话号码的字母组合](#三、17. 电话号码的字母组合)
      • [四、39. 组合总和](#四、39. 组合总和)
      • [五、40. 组合总和 II](#五、40. 组合总和 II)

一、77. 组合

题目链接:https://leetcode.cn/problems/combinations/description/

思路:元素无重,不可复用,求组合数,可以早停。常规做法套模板。

java 复制代码
class Solution {
    List<List<Integer>> resList = new ArrayList<>();
    List<Integer> list = new ArrayList<>();
    public List<List<Integer>> combine(int n, int k) {
        backTracking(n, k, 1);
        return resList;
    }
    
    void backTracking(int n, int k, int start) {
        if(list.size() == k) {
            resList.add(new ArrayList(list));
            return;
        }
        for(int i = start; i <= n && n - (k - list.size()) + 1 >= i; i++) {
            list.add(i);
            backTracking(n, k, i+1);
            list.remove(list.size()-1);
        }
    }
}

二、216. 组合总和 III

题目链接:https://leetcode.cn/problems/combination-sum-iii/description/

思路:元素无重,不可复选,要求和为n的k个数的组合,只需要简单的判断和早停,其他套模板。

java 复制代码
class Solution {
    List<List<Integer>> resList = new ArrayList<>();
    List<Integer> list = new ArrayList<>();
    int sum = 0;
    public List<List<Integer>> combinationSum3(int k, int n) {
        if(k > n) return resList;
        backTracking(k, n, 1);
        return resList;
    }

    void backTracking(int k, int n, int start) {
        if(sum == n && list.size() == k) {
            resList.add(new ArrayList(list));
            return;
        }
        for(int i = start; i <= 9 && sum + i <= n; i++) {
            list.add(i);
            sum += i;
            backTracking(k, n, i+1);
            sum -= i;
            list.remove(list.size()-1);
        }
    }
    
}

三、17. 电话号码的字母组合

题目链接:https://leetcode.cn/problems/letter-combinations-of-a-phone-number/description/

思路:集合无重,元素不可复用,本题不同的点是所使用的集合是可以变化的,每次向下递归需要更换下一个集合。

java 复制代码
class Solution {
    List<String> list = new ArrayList<>();
    StringBuilder builder = new StringBuilder();
    String[] source = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
    public List<String> letterCombinations(String digits) {
        if(digits.equals("")) return list;
        backTracking(0, digits);
        return list;
    }
    
    void backTracking(int start, String digits) {
        if(builder.length() == digits.length()) {
            list.add(builder.toString());
            return;
        }
        String temp = source[digits.charAt(start) - '0'];
        for(int i = 0; i < temp.length(); i++) {
            builder.append(temp.charAt(i));
            backTracking(start+1, digits);
            builder.deleteCharAt(builder.length()-1);
        }
    }
}

四、39. 组合总和

题目链接:https://leetcode.cn/problems/combination-sum/description/

思路:集合无重,元素可复选,求和为K的组合数,向下递归索引位置为i不用加1,确保单个元素可以复选,排序后可以早停。其他套模板。

java 复制代码
class Solution {
    List<List<Integer>> resList = new ArrayList<>();
    List<Integer> list = new ArrayList<>();
    int sum = 0;
    public List<List<Integer>> combinationSum(int[] candidates, int target) {
        Arrays.sort(candidates);
        backTracking(candidates, target, 0);
        return resList;
    }
    
    void backTracking(int[] candidates, int target, int start) {
        if(sum == target) {
            resList.add(new ArrayList(list));
            return;
        }
        for(int i = start; i < candidates.length && sum + candidates[i] <= target; i++) {
            sum += candidates[i];
            list.add(candidates[i]);
            backTracking(candidates, target, i);
            sum -= candidates[i];
            list.remove(list.size()-1);
        }
    }
    
}

五、40. 组合总和 II

题目链接:https://leetcode.cn/problems/combination-sum-ii/description/

思路:集合元素有重,不可复选,求和为K的组合数,数组和排序,排序后,纵向不去重,横向去重,横向只需要大于初始索引,前后两个值想对,即去重。其他一样。

java 复制代码
class Solution {
    List<List<Integer>> resList = new ArrayList<>();
    List<Integer> list = new ArrayList<>();
    int sum = 0;
    public List<List<Integer>> combinationSum2(int[] candidates, int target) {
        Arrays.sort(candidates);
        backTracking(candidates, target, 0);
        return resList;
    }

    void backTracking(int[] candidates, int target, int start) {
        if(sum == target) {
            resList.add(new ArrayList(list));
            return;
        }
        for(int i = start; i < candidates.length && sum + candidates[i] <= target; i++) {
            if(i > start && candidates[i] == candidates[i-1]) continue;
            sum += candidates[i];
            list.add(candidates[i]);
            backTracking(candidates, target, i+1);
            sum -= candidates[i];
            list.remove(list.size()-1);
        }
    }
    
}
相关推荐
ChoSeitaku21 分钟前
链表循环及差集相关算法题|判断循环双链表是否对称|两循环单链表合并成循环链表|使双向循环链表有序|单循环链表改双向循环链表|两链表的差集(C)
c语言·算法·链表
DdddJMs__13526 分钟前
C语言 | Leetcode C语言题解之第557题反转字符串中的单词III
c语言·leetcode·题解
Fuxiao___30 分钟前
不使用递归的决策树生成算法
算法
我爱工作&工作love我35 分钟前
1435:【例题3】曲线 一本通 代替三分
c++·算法
白-胖-子1 小时前
【蓝桥等考C++真题】蓝桥杯等级考试C++组第13级L13真题原题(含答案)-统计数字
开发语言·c++·算法·蓝桥杯·等考·13级
workflower1 小时前
数据结构练习题和答案
数据结构·算法·链表·线性回归
好睡凯1 小时前
c++写一个死锁并且自己解锁
开发语言·c++·算法
Sunyanhui11 小时前
力扣 二叉树的直径-543
算法·leetcode·职场和发展
一个不喜欢and不会代码的码农1 小时前
力扣105:从先序和中序序列构造二叉树
数据结构·算法·leetcode
前端郭德纲2 小时前
浏览器是加载ES6模块的?
javascript·算法