【Leetcode】77 组合 | 掌握回溯的力量吧!

news2024/12/26 16:07:45

【1】限制:数字只能够使用一次。

77 组合

栗子,从 { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 } \{1,2,3,4,5,6,7,8,9,10\} {1,2,3,4,5,6,7,8,9,10}中选择4个数:

  • 选择1,从 { 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 } \{2,3,4,5,6,7,8,9,10\} {2,3,4,5,6,7,8,9,10}中选择3个数;
  • 选择2,从 { 3 , 4 , 5 , 6 , 7 , 8 , 9 , 10 } \{3,4,5,6,7,8,9,10\} {3,4,5,6,7,8,9,10}中选择3个数;
  • 选择3,从 { 4 , 5 , 6 , 7 , 8 , 9 , 10 } \{4,5,6,7,8,9,10\} {4,5,6,7,8,9,10}中选择3个数;

… \dots

  • 选择7,从 8 , 9 , 10 {8,9,10} 8,9,10中选择3个数。
  • 选择8,将不足4个数,give up~

image.png

很好,nice,怎么实现呢?

  • 叶子结点的信息体现在从根结点到叶子结点的路径上,因此需要一个表示路径的全局变量 pathpath 是一个栈;
  • 同时,需要一个全局变量res存储返回结果;
  • DFS(int start, int n, int k)表示在[start, n]区间,选取k个数。
import java.util.ArrayList;
import java.util.List;
import java.util.Stack;

class Solution {

    // 用于表示经过的路径
    Stack<Integer> path = new Stack<>();
    // 用于返回结果
    List<List<Integer>> res = new ArrayList<>();

    public List<List<Integer>> combine(int n, int k) {
        DFS(1, n, k);
        return res;
    }

    public void DFS(int start, int n, int k) {
        // 在[start, n]区间,选取k个数
        // 递归终止条件
        if (k == 0) {
            res.add(new ArrayList<>(path));
            return;
        }
        // i最远到[n - k + 1, n],此时长度为k
        for (int i = start; i <= n - k + 1; i++) {
            // 选择i
            path.push(i);
            // 再从[i + 1, n]选k-1个数
            DFS(i + 1, n, k - 1);
            // 回溯
            path.pop();
        }
    }

}

216 组合总和Ⅲ

类似的思路,不过可以进行一些剪枝:

  • n > 45,必找不到满足的组合
  • DFS(int start, int k, int n) :在[start, 9]中选择k个数,使其和为n
  • 递归终止
    • 找到满意的组合:(k == 0 && n == 0)
    • 找不到满意的组合了:背包容量不足 || 可用数字额度已用完,但包未装满(n < 0 || k == 0)
import java.util.ArrayList;
import java.util.List;
import java.util.Stack;

class Solution {
    Stack<Integer> path = new Stack<>();
    List<List<Integer>> res = new ArrayList<>();

    public List<List<Integer>> combinationSum3(int k, int n) {
        // 题目描述:找到1-9中的k组合,满足相加等于n
        // 为了不重不漏,还是用相同的方法遍历,含1,不含1含2,……
        if (n > 45) {
            return new ArrayList<>();
        }
        DFS(1, k, n);
        return res;
    }

    public void DFS(int start, int k, int n) {
        // 函数含义:在[start, 9]中选择k个数,使其和为n

        // 1、如果找到了满足的组合
        if (k == 0 && n == 0) {
            res.add(new ArrayList<>(path));
            return;
        }

        // 如果n小于0,说明不可能找到满足的组合啦
        // 如果n大于0且k等于0,没有可用的数字了,但n还没有被消耗完
        if (n < 0 || k == 0) {
            return;
        }

        // 要保证可找范围大于k个数
        // i最大为[10 - K, 9]
        for (int i = start; i <= 10 - k; i++) {
            path.push(i);
            DFS(i + 1, k - 1, n - i);
            path.pop();
        }
    }
}

17 电话号码的字母组合

遍历完整个字符串就可以辣。

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;

class Solution {
    HashMap<Character, char[]> phone = new HashMap<Character, char[]>() {{
        put('2', new char[]{'a', 'b', 'c'});
        put('3', new char[]{'d', 'e', 'f'});
        put('4', new char[]{'g', 'h', 'i'});
        put('5', new char[]{'j', 'k', 'l'});
        put('6', new char[]{'m', 'n', 'o'});
        put('7', new char[]{'p', 'q', 'r', 's'});
        put('8', new char[]{'t', 'u', 'v'});
        put('9', new char[]{'w', 'x', 'y', 'z'});
    }};

    StringBuilder path = new StringBuilder();
    List<String> res = new ArrayList<>();

    public List<String> letterCombinations(String digits) {
        if ("".equals(digits)) {
            return res;
        }
        DFS(digits.toCharArray(), 0);
        return res;
    }

    public void DFS(char[] digits, int i) {
        // 从第i个字母开始的digits数组,可以获得多少种ans
        if (i == digits.length) {
            res.add(path.toString());
            return;
        }
        char num = digits[i];
        for (char alpha: phone.get(num)) {
            path.append(alpha);
            DFS(digits, i + 1);
            path.deleteCharAt(i);
        }
    }
}

39 组合总和

【2】限制:数字可以重复使用,但不能生成重复组合。

回溯树的样子有所改变:

39.组合总和

import java.util.ArrayList;
import java.util.List;
import java.util.Stack;

class Solution {
    Stack<Integer> path = new Stack<>();
    List<List<Integer>> res = new ArrayList<>();

    public List<List<Integer>> combinationSum(int[] candidates, int target) {
        // candidates不重复
        DFS(candidates, 0, target);
        return res;
    }

    public void DFS(int[] candidates, int start, int target) {
        // 可选candidates[start:]的数字,组合为target
        if (target == 0) {
            res.add(new ArrayList<>(path));
            return;
        }
        if (target < 0 || start == candidates.length) {
            return;
        }

        // 如果选择该数字的话,下一步仍可选择该位置数字
        path.push(candidates[start]);
        DFS(candidates, start, target - candidates[start]);
        path.pop();
        // 不选择该位置数字的话,下一步就要往下移动
        DFS(candidates, start + 1, target);
    }
}

40 组合总和Ⅱ★

【3】限制:数字可以重复使用,但不能生成重复组合。

提供的可选数组列表有重复。

根据上一题可知,应该限制同一个数字的树枝最多向下 f r e q freq freq次。

采用和三数之和类似的思路,第一个元素应该是不同的元素。

  • 剪枝:若candidates[i] > target,之后的情况可直接不用考虑。

  • 去重:

    // 如果不是第一个start位置的数字
    // 并且和前一个数字相等,由于前一个数字已经向下扩展过了,这个树枝就不必再扩展哩(参考三数之和思路
    if (i > start && candidates[i] == candidates[i - 1]) {
        continue;
    }
    
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Stack;

class Solution {
    Stack<Integer> combination = new Stack<>();
    List<List<Integer>> res = new ArrayList<>();

    public List<List<Integer>> combinationSum2(int[] candidates, int target) {
        Arrays.sort(candidates);
        DFS(candidates, 0, target);
        return res;
    }

    public void DFS (int[] candidates, int start, int target) {
        // 含义:从candidates[start:]找到和为target的组合

        // 如果target降到0,说明找到了这样的组合
        if (target == 0) {
            res.add(new ArrayList<>(combination));
            return;
        }

        // 如果target < 0或者走到了尽头, 剪枝
        if (target < 0 || start == candidates.length) {
            return;
        }

        // 这个for循环代表了同一层的递归树,
        // 如果遍历到某个值时大于target,不必再向下考虑,剪枝
        for (int i = start; i < candidates.length && candidates[i] <= target; i++) {
            
            // 如果和前一个数字相等,由于前一个数字已经向下扩展过了,这个树枝就不必再扩展哩
            if (i > start && candidates[i] == candidates[i - 1]) {
                continue;
            }

            // 选择start位置
            combination.push(candidates[i]);
            DFS(candidates, i + 1, target - candidates[i]);
            combination.pop();
        }

    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/617655.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

电力需求侧管理和电力负荷管理数字化解决方案 安科瑞 许敏

摘要&#xff1a;近年来全国用电负荷特别是居民用电负荷的快速增长&#xff0c;全国范围内夏季、冬季用电负荷“双峰”特征日益突出&#xff0c;恶劣气候现象多发增加了电力安全供应的压力。具有随机性、波动性、间歇性特征的可再生能源大规模接入电网对电力系统的稳定性带来新…

视频观看行为高级分析(大数据分析)

今天介绍一下我们的视频观看行为高级分析功能。 一、观看行为分析 观看行为分析&#xff0c;基于Polyv大数据分析&#xff0c;能够以秒为粒度展示观众如何观看您的视频。 视频观看热力图是单次观看行为的图形化表示&#xff0c;Polyv云点播视频的每一次播放&#xff0c;都会产…

基于jupyter的多分类问题练习

文章目录 练习3&#xff1a;多分类问题介绍1 多分类1.1 数据集1.2 数据可视化1.3 逻辑回归的向量化1.3.1 代价函数的向量化1.3.2 梯度的向量化1.3.3 正则化逻辑回归的向量化 1.4 多分类-分类器 1.5 使用分类器进行预测 总结 练习3&#xff1a;多分类问题 介绍 在本练习中&…

Leetcode周赛348

第一题&#xff1a;最小化字符串长度 思路分析 通过分析我们可以发现&#xff0c;只要存在重复的元素就可以继续进行操作所以这里本质上是一道去重的题目去重我们可以使用双指针算法和Set&#xff1b;我们选择使用Set进行去重 class Solution {public int minimizedStringLengt…

Vue2 vue-cli

安装与卸载vue脚手架 npm i -g vue/cli vue --version 查看vue脚手架版本 vue -V 查看vue脚手架版本 npm uninstall -g vue/cli 卸载 创建项目 vue create 项目名 选择项目 &#xff08;Default 为快速创建项目&#xff09; 选择最后一下&#xff0c;回车 上下键选择 Rou…

shell脚本:函数

shell脚本-函数 一、函数&#xff1a;1.定义&#xff1a;2.作用&#xff1a;3.格式&#xff1a; 二、函数传参&#xff1a;1.定义&#xff1a;2.函数变量&#xff1a;3.递归&#xff1a;4.函数库&#xff1a; 一、函数&#xff1a; 1.定义&#xff1a; &#xff08;1&#xf…

Internal error. Please report to https://jb.gg/ide/critical-startup-errors

大佬的解决方式&#xff1a;PyCharm 2023 启动报错的处理 部分同学&#xff0c;发现在安装 PyCharm 2023.1.2 以及 PyCharm 2023.2 的抢先体验版之后&#xff0c;运行的时候愣是直接弹出了类似上面的报错。 反正&#xff0c;别慌&#xff01; 是的&#xff0c;他们有 bug。 …

呈现视觉妙技:使用Python将MP4视频转化为迷人的GIF图像

前言 GIF图片对于我来说是一个很好的展示方式&#xff0c;GIF 图片能够展示动态的图像效果&#xff0c;对于展示计算机视觉算法或结果非常有用。例如&#xff0c;我可以使用 GIF 图片来展示运动跟踪、姿势识别、图像分割、目标检测等任务的结果&#xff0c;以更生动和直观的方…

基于midiepipe、opencv的家庭健身智能推荐与姿态监测系统

目录 0. 前言1. opencv简介2. midiepipe简介3. yolo5简介4. 虚拟环境搭建以及工程目录设置5. 部分代码展示6. 项目成果7. 说明 0. 前言 本项目由我与gay友以及三位21级学弟历时一年共同合作完成 为了给运动者提供一种更加科学、更加精准的个性化运动方案&#xff0c;设计实现…

【SpinalHDL快速入门】6.2、SpinalHDL语法之When/Switch/Mux

文章目录 1.1、When1.2、Switch1.2.1、实例1.2.2、附加选项 1.3、本地声明1.4、Mux1.5、位选择1.5.1、实例 1.1、When 与VHDL和Verilog一样&#xff0c;当满足指定条件时可以对信号进行有条件的赋值&#xff1a; when(cond1) {// Execute when cond1 is true }.elsewhen(cond…

直播问答功能(互动功能接收端JS-SDK)

功能概述 本模块主要用于展示问答模块。 初始化及销毁 在实例化该模块并进行使用之前&#xff0c;需要对SDK进行初始化配置&#xff0c;详细见参考文档。 在线文件引入方式 // script 标签引入&#xff0c;根据版本号引入JS版本。 <script src"https://websdk.vi…

【Maven】Maven入门,Java界的Anaconda!

1 Maven介绍 官方文档&#xff1a;Apache Maven 的本质是一个软件项目管理和理解工具。基于项目对象模型 (Project Object Model&#xff0c;POM) 的概念&#xff0c;Maven 可以从一条中心信息管理项目的构建、报告和文档。 就像Python和Anaconda的关系&#x1f92d; 可以帮助…

java设计模式(十三)解释器模式

目录 定义模式结构角色职责代码实现场景适用优缺点 定义 解释器模式&#xff08;Interpreter Pattern&#xff09; 提供了评估语言的语法或表达式的方式&#xff0c;它属于行为型模式。这种模式实现了一个表达式接口&#xff0c;该接口解释一个特定的上下文。这种模式被用在 S…

Linux 内存管理6——slab内存池的创建初始化过程

在上篇文章 中&#xff0c;笔者带大家从一个最简单的物理内存页开始&#xff0c;一步一步演进 slab cache 的架构&#xff0c;最终得到了一副 slab cache 完整的架构图&#xff1a; 在本文的内容中&#xff0c;笔者会带大家到内核源码实现中&#xff0c;来看一下 slab cache 在…

第三章:基本的SELECT语句

第三章&#xff1a;基本的SELECT语句 3.1&#xff1a;SQL概述 SQL背景知识 1946年&#xff0c;世界上第一台电脑诞生&#xff0c;如今&#xff0c;借由这台电脑发展起来的互联网已经自成江湖。在这几十年里&#xff0c;无数的技术、产业在这片江湖里沉浮&#xff0c;有的方兴未…

HTTP和HTTPS

目录 HTTP 告知服务器意图的HTTP方法 状态码告知从服务器端返回的请求结果 2XX 成功 3XX 重定向 4XX客户端错误 5XX服务器错误 使用Cookie的状态管理 HTTP 的缺点 HTTPS HTTP 当我们在网页浏览器的地址栏输入URL时&#xff0c;Web页面是如何呈现的吗&#xff1f; …

Linux 面试题-(腾讯,百度,美团,滴滴)

Linux 面试题-(腾讯,百度,美团,滴滴) 分析日志t.log(访问量)&#xff0c;将各个ip 地址截取&#xff0c;并统计出现次数,并按从大到小排序(腾讯) http://192.168.200.10/index1.html http://192.168.200.10/index2.html http://192.168.200.20/index1.html http://192.168.20…

Python 类和对象

一、什么是类和对象 Python和Java一样&#xff0c;都是面向对象的编程语言&#xff0c;面向对象编程其实是一种封装代码的方法&#xff0c;把一些公共的属性或者方法封装到一个类中&#xff0c;然后再通过这个类可以创建多个对象&#xff0c;最后使用这些对象去调用这些封装起…

【教程】两种免费更新iOS17测试版的方法

苹果iOS17系统已经发布&#xff0c;目前所有用户都可以免费注册成为开发者&#xff0c;升级iOS17开发者测试版 注意&#xff0c;现在不是通过描述文件来更新系统了&#xff0c;给大家带来两种更新升级方法&#xff0c;看下文操作 方法一 苹果官网注册 按照下图发消息“更新” …

java设计模式(十六)命令模式

目录 定义模式结构角色职责代码实现适用场景优缺点 定义 命令模式&#xff08;Command Pattern&#xff09; 又叫动作模式或事务模式。指的是将一个请求封装成一个对象&#xff0c;使发出请求的责任和执行请求的责任分割开&#xff0c;然后可以使用不同的请求把客户端参数化&a…