代码随想录第25天|216.组合总和III 17.电话号码的字母组合

216.组合总和III 

216. 组合总和 III - 力扣(LeetCode)

代码随想录 (programmercarl.com)

和组合问题有啥区别?回溯算法如何剪枝?| LeetCode:216.组合总和III_哔哩哔哩_bilibili

找出所有相加之和为 n 的 k 个数的组合,且满足下列条件:

  • 只使用数字1到9
  • 每个数字 最多使用一次 

返回 所有可能的有效组合的列表 。该列表不能包含相同的组合两次,组合可以以任何顺序返回。

示例 1:

输入: k = 3, n = 7
输出: [[1,2,4]]
解释:
1 + 2 + 4 = 7
没有其他符合的组合了。

示例 2:

输入: k = 3, n = 9
输出: [[1,2,6], [1,3,5], [2,3,4]]
解释:
1 + 2 + 6 = 9
1 + 3 + 5 = 9
2 + 3 + 4 = 9
没有其他符合的组合了。

示例 3:

输入: k = 4, n = 1
输出: []
解释: 不存在有效的组合。
在[1,9]范围内使用4个不同的数字,我们可以得到的最小和是1+2+3+4 = 10,因为10 > 1,没有有效的组合。

第一反应和昨天的77.组合的解法相同,不过对于剪枝部分以及n和k怎么应用我还是有点不懂。

看了卡哥的题解,发现自己对于题目的理解还是有些偏差。首先,该题题意可以理解为:在[1,2,3,4,5,6,7,8,9]这个集合中找到和为n的k个数的组合。K相当于树的深度。

回溯三部曲

1、确定参数和返回值:一个一维数组path来存放符合条件的结果,一个二位数组result来存放结果集:

List<List<Integer>> result = new ArrayList<>(); // 用于存储最终结果的列表
LinkedList<Integer> path = new LinkedList<>(); // 用于存储当前组合的列表

还需要的参数有:n,k,path路径中元素的综合sum,下一层for循环搜索的起始位置startIndex:
 

private void backTracking(int targetSum, int k, int startIndex, int sum) {
   
}

2、确定终止条件:path.size和k相等的时候终止,sum和targetSum相等的时候用result收集当前结果:

if (path.size() == k) {
    if (sum == targetSum) 
    result.add(new ArrayList<>(path));
    return;
}

3、单层搜索的过程:

for (int i = startIndex; i <= 9 ; i++) {
    path.add(i); // 将当前数字加入组合列表
    sum += i; // 更新累加和
    backTracking(targetSum, k, i + 1, sum); // 递归搜索下一个数字
    path.removeLast(); // 回溯:移除最后加入的数字,以便尝试其他可能的数字
    sum -= i; // 回溯:减去移除的数字,恢复累加和
}

 剪枝:

1、已选元素的值已经大于总和,就没有必要再往后遍历了

2、for循环的位置可以剪枝:i <= 9 - (k - path.size()) + 1就可以了。

综合代码

class Solution {
    List<List<Integer>> result = new ArrayList<>(); // 用于存储最终结果的列表
    LinkedList<Integer> path = new LinkedList<>(); // 用于存储当前组合的列表

    public List<List<Integer>> combinationSum3(int k, int n) {
        backTracking(n, k, 1, 0); // 调用回溯算法寻找符合条件的组合
        return result; // 返回最终结果
    }

    private void backTracking(int targetSum, int k, int startIndex, int sum) {
        // 减枝:如果当前累加的和已经大于目标值,则停止当前路径的搜索
        if (sum > targetSum) {
            return;
        }

        // 如果当前组合的元素数量已经达到要求,且当前累加和等于目标值,将当前组合加入结果列表
        if (path.size() == k) {
            if (sum == targetSum) 
                result.add(new ArrayList<>(path));
            return;
        }

        // 减枝:根据剩余可用数字的数量和目标组合元素数量,确定可用的起始数字范围
        for (int i = startIndex; i <= 9 - (k - path.size()) + 1; i++) {
            path.add(i); // 将当前数字加入组合列表
            sum += i; // 更新累加和
            backTracking(targetSum, k, i + 1, sum); // 递归搜索下一个数字
            path.removeLast(); // 回溯:移除最后加入的数字,以便尝试其他可能的数字
            sum -= i; // 回溯:减去移除的数字,恢复累加和
        }
    }
}

 17.电话号码的字母组合 

17. 电话号码的字母组合 - 力扣(LeetCode)

代码随想录 (programmercarl.com)

还得用回溯算法!| LeetCode:17.电话号码的字母组合_哔哩哔哩_bilibili

有了上一题的基础和思路之后,明白这一题要用回溯,也知道树怎么画,但是对于代码怎么实现还是有些不太明白。

看了卡哥视频题解:

从图中可以看出,遍历的深度就是【2、3】数组的长度,叶子节点就是我们要收集的结果。

回溯三部曲

1、确定参数:全局变量:一个字符串S手机叶子节点的结果,然后用一个字符串数组result把结果收集起来;局部变量:string digits和index,index用来遍历digits的同时记录遍历到digits的第几个数字;

2、确定终止条件:index=digits.size

3、确定单层遍历的逻辑:取index指向的数字,取该数字对应的字母:提前把index中的数字和其对应的字母集用map存储起来

综合代码

class Solution {

    // 设置全局列表存储最后的结果
    List<String> list = new ArrayList<>();

    // 主函数,传入数字字符串,返回所有可能的字母组合
    public List<String> letterCombinations(String digits) {
        if (digits == null || digits.length() == 0) {
            return list; // 如果输入为空,则直接返回结果列表
        }
        
        // 初始对应所有的数字,为了直接对应2-9,新增了两个无效的字符串""
        String[] numString = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
        
        // 迭代处理
        backTracking(digits, numString, 0);
        return list;
    }

    // 每次迭代获取一个字符串,所以会设计大量的字符串拼接,所以这里选择更为高效的 StringBuilder
    StringBuilder temp = new StringBuilder();

    // 比如 digits 如果为 "23",num 为 0,则 str 表示 2 对应的 "abc"
    public void backTracking(String digits, String[] numString, int num) {
        // 遍历全部一次记录一次得到的字符串
        if (num == digits.length()) {
            list.add(temp.toString()); // 当数字索引已经达到 digits 字符串的长度时,将当前拼接好的字符串加入结果列表
            return;
        }
        
        // str 表示当前 num 对应的字符串
        String str = numString[digits.charAt(num) - '0'];
        for (int i = 0; i < str.length(); i++) {
            temp.append(str.charAt(i)); // 将当前字符追加到临时 StringBuilder 中
            backTracking(digits, numString, num + 1); // 递归调用,处理下一个数字
            temp.deleteCharAt(temp.length() - 1); // 剔除末尾的字符,为下一个迭代做准备
        }
    }
}


 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于自动编码器的预训练模型方法模型预训练方法RetroMAE和RetroMAE-2

文章目录 RetroMAERetroMAE详情编码解码增强解码 RetroMAE-2RetroMAE-2详情编码[CLS]解码OT解码和训练目标向量表征 总结参考资料 RetroMAE RetroMAE 出自论文《RetroMAE: Pre-Training Retrieval-oriented Language Models Via Masked Auto-Encoder》&#xff0c;是一种针对于…

「MySQL」索引事务

&#x1f387;个人主页&#xff1a;Ice_Sugar_7 &#x1f387;所属专栏&#xff1a;数据库 &#x1f387;欢迎点赞收藏加关注哦&#xff01; 索引&事务 &#x1f349;索引&#x1f34c;特点&#x1f34c;通过 SQL 操作索引&#x1f34c;底层数据结构 &#x1f349;事务&…

网络编程的学习1

网络编程 在网络通信协议下&#xff0c;不同计算机上运行的程序&#xff0c;进行数据传输。 三要素 ip&#xff1a;设备在网络中的地址&#xff0c;是唯一的标识。 ipv4:采取32位地址长度&#xff0c;分成4组。 ipv6&#xff1a;采用128位地址长度&#xff0c;分成8组。 …

安卓SharedPreferences使用

目录 一、简介二、使用2.1 getSharedPreferences2.2 增加数据2.3 读取数据2.4 删除数据2.5 修改数据2.6 清除数据2.7 提交数据 一、简介 SharedPreferences是Android平台上一个轻量级的存储类&#xff0c;主要是保存一些常用的配置比如窗口状态&#xff0c;一般在Activity、重…

12.Python文件读写

文件是数据的载体&#xff0c;程序可以从文件中读取数据&#xff0c;也可以将数据写 入文件中&#xff0c;本章重点介绍如何在Python中进行文件读写。 1 打开文件 我们在使用文件之前要先将文件打开&#xff0c;这通过open&#xff08;&#xff09;函数实现。 open&#xff0…

JJJ:linux系统中第一个进程

以linux4.19内核linux系统中第一个进程。 执行shell指令 ps -ef 结果如下&#xff1a; xxxxxx-virtual-machine:~$ ps -ef UID PID PPID C STIME TTY TIME CMD root 1 0 0 20:55 ? 00:00:02 /sbin/init splash root …

【Web应用技术基础】CSS(4)——背景样式

第1题&#xff1a;背景颜色 .html <!DOCTYPE html> <html><head><meta charset"utf-8"><title>Hello World</title><link rel"stylesheet" href"step1/CSS/style.css"> </head><body>&…

词令蚂蚁庄园今日答案正确答案怎么获取查看蚂蚁庄园今天问题的正确答案?

词令蚂蚁庄园今日答案正确答案怎么获取查看蚂蚁庄园今天问题的正确答案&#xff1f; 1、打开「词令」关键词口令直达工具&#xff1b; 2、输入词令关键词直达口令「今日答案999」&#xff1b; 3、搜索直达词令蚂蚁庄园今日问题的正确答案&#xff1b; *注&#xff1a;词令蚂蚁…

【PythonGIS】Python实现批量导出面矢量要素(单个多面矢量->多个单面矢量)

可怜的我周六还在工作&#xff0c;已经很久没更新过博客了&#xff0c;今天正好有空就和大家分享一下。今天给大家带来的是使用Python将包含多个面要素/线要素的矢量批量导出单个要素的矢量&#xff0c;即一个要素一个矢量文件。之前写过多个矢量文件合并成一个矢量文件的博文&…

西南交大swjtu算法实验3.3|穷举法

1.实验目的 通过具体例子学习排列这种典型的穷举算法的求解过程以及程序框架&#xff0c;分析其算法的求解过程&#xff0c;以及如何设计穷举法解决实际问题。通过本实验&#xff0c;理解穷举法的特点以及实际应用中的局限性。 2.实验任务 有n (n>1&#xff09;个任务需要…

Visual Studio 2022 中VLD库如何安装

GitHub链接 Release v2.5.1 KindDragon/vld 点击可执行程序进行下载 点击可执行程序进行安装 双击打开 一直点击next即可完成安装&#xff08;不用在意安装路径&#xff0c;总共不到2MB&#xff09; 如果GitHub无法打开&#xff0c;可以私信我发你安装包直接安装

fpga_awb

色温: sesor原始图像中的白色如果不经AWB处理&#xff0c;在高色温(如阴天)下偏蓝&#xff0c;在低色温下偏黄。 引入白平衡算法 而AWB的核心就是调整图像色温&#xff0c;使得摄像头采集的图像更加真实&#xff0c;达到人眼观察的效果。 白平衡一般通过调节传感器输出图像RGB…

【aws】架构图工具推荐

碎碎念 以前以为日本冰箱论是个梗&#xff0c;结果居然是真的。用光盘传真其实还能理解&#xff08;毕竟我也喜欢电子古董2333&#xff09;&#xff0c;但是画架构图居然用的是excel&#xff0b;截图&#xff01;啊苍天呐&#xff0c;然后看到隔壁工位用excel画web原型又感觉释…

svg实现环形进度条

实现效果图&#xff1a; svg相关知识 这里只介绍本次用到的元素&#xff0c;更多详情&#xff1a;SVG&#xff1a;可缩放矢量图形 defs&#xff1a;定义需要重复利用的图形元素linearGradient&#xff1a;定义线性渐变&#xff0c;用来图形元素的填充或描边使用stop&#x…

hcip综合实验2

目录 实验拓扑&#xff1a; 实验要求&#xff1a; 实验思路&#xff1a; 实验步骤&#xff1a; 1.配置设备接口IP 2.通过配置缺省路由让公网互通 3.配置ppp 1.R1和R5间的ppp的PAP认证&#xff1b; 2.R2与R5之间的ppp的CHAP认证; 3. R3与R5之间的HDLC封装; 4.构建R1、…

动态规划之子序列(三)

583. 两个字符串的删除操作 给定两个单词 word1 和 word2&#xff0c;找到使得 word1 和 word2 相同所需的最小步数&#xff0c;每步可以删除任意一个字符串中的一个字符。 示例&#xff1a; 输入: “sea”, “eat” 输出: 2 解释: 第一步将"sea"变为"ea"…

c实现猜数游戏(猜不对可是要自动帮你电脑关机)

接下来的日子会顺顺利利&#xff0c;万事胜意&#xff0c;生活明朗-----------林辞忧 前言 猜数字游戏作为一个基础的C程序小项目&#xff0c;实现简单&#xff0c;可以帮助我们巩固很多知识&#xff0c;作为扩展接下来我们实现一个自定猜数次数&#xff0c;用完次数电脑自动…

keepalived+LVS高可用部署

目录 一.两台设备&#xff08;2.130和2.133&#xff09;作为调度器&#xff0c;前主后备 1.部署keepalived 2.修改配置文件准备启动 3.配置keepalived的系统日志并启动 二.模拟调度器掉点和web服务进程丢失 1.调度器掉点 2.当类似于httpd这种网站服务掉点 三.以三种健康…

【从前端入门到全栈】前端框架之核心概念

大家好&#xff0c;我是江辰&#xff0c;从前端入门到全栈是我全新系列文章&#xff0c;从去年一直囔囔着要写&#xff0c;今年总算开始了&#xff01;预计在10篇左右。知识面从 前端&#xff0c;后端&#xff0c;运维&#xff0c;脚本等&#xff0c;都有涉及&#xff0c;主打一…

Spark-Scala语言实战(9)

之前的文章中&#xff0c;我们学习了如何在spark中使用RDD方法的flatMap,take,union。想了解的朋友可以查看这篇文章。同时&#xff0c;希望我的文章能帮助到你&#xff0c;如果觉得我的文章写的不错&#xff0c;请留下你宝贵的点赞&#xff0c;谢谢。 Spark-Scala语言实战&am…