24暑假算法刷题 | Day22 | LeetCode 77. 组合,216. 组合总和 III,17. 电话号码的字母组合

目录

  • 77. 组合
    • 题目描述
    • 题解
  • 216. 组合总和 III
    • 题目描述
    • 题解
  • 17. 电话号码的字母组合
    • 题目描述
    • 题解


77. 组合

点此跳转题目链接

题目描述

给定两个整数 nk,返回范围 [1, n] 中所有可能的 k 个数的组合。

你可以按 任何顺序 返回答案。

示例 1:

输入:n = 4, k = 2
输出:
[[2,4],[3,4],[2,3],[1,2],[1,3],[1,4],
]

示例 2:

输入:n = 1, k = 1
输出:[[1]]

提示:

  • 1 <= n <= 20
  • 1 <= k <= n

题解

参考:代码随想录-77.组合 ,讲得非常细了。

算是回溯算法的入门题目,核心要理解回溯的树形结构,以及其中的横向纵向遍历逻辑:

img

然后,考虑回溯三部曲

1️⃣ 处理

本题就是很基础的求组合,所以每次往当前组合 path 添加新数字就好了:

path.push_back(i); 

当前组合名称取为 path ,旨在呼应回溯树形结构图中的纵向“探索”路线(先取一个数x,再取一个数y)

2️⃣ 递归

递归出口是经典的——当前组合大小达到目标组合大小,则将其加入结果集:

if (path.size() == k)
{res.push_back(path);return;
}

否则,当前位置数字确定后,递归下一个位置的数来做组合:

backTracking(i + 1, end, k)

3️⃣ 回溯

弹出当前组合的最后一个值,以便探索该位置的其他可能值:

path.pop_back();   

整体代码如下:

C++

class Solution
{
private:vector<int> path;vector<vector<int>> res;public:void backTracking(int start, int end, int k){// 回溯出口:子结果path已满(纵向遍历)if (path.size() == k){res.push_back(path);return;}// 横向遍历for (int i = start; i <= end; i++){path.push_back(i);           // 处理backTracking(i + 1, end, k); // 递归path.pop_back();             // 回溯}}vector<vector<int>> combine(int n, int k){backTracking(1, n, k);return res;}
};

Go

type Helper struct {path []intres  [][]int
}func (helper *Helper) backTracking(start int, end int, k int) {// 递归出口if len(helper.path) == k {// newPath := make([]int, len(helper.path))// copy(newPath, helper.path)// helper.res = append(helper.res, newPath)helper.res = append(helper.res, append([]int(nil), helper.path...))return}// 横向遍历for i := start; i <= end; i++ {helper.path = append(helper.path, i)           // 处理helper.backTracking(i+1, end, k)               // 递归helper.path = helper.path[:len(helper.path)-1] // 回溯}
}func combine(n int, k int) [][]int {helper := Helper{}helper.backTracking(1, n, k)return helper.res
}

216. 组合总和 III

点此跳转题目链接

题目描述

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

  • 只使用数字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,没有有效的组合。

提示:

  • 2 <= k <= 9
  • 1 <= n <= 60

题解

参考:代码随想录-216

回溯算法解决。首先想清楚回溯的树形结构图:

在这里插入图片描述

然后走回溯三部曲

  • 处理: 将当前处理的数字加入当前组合 path ,并求此时组合中的数字和

    进行这一步之前可以剪枝:由于是从1到9(从小到大)横向遍历,如果 目标和 与 当前和 的差小于即将加入的数字,说明再加数字必将导致组合总和过大,故没必要在此基础上往后遍历处理了。

  • 递归: 递归地尝试将后面的数字加入组合;递归出口: path 的大小达到目标大小 k ,且其中数字和等于目标和,则将 path 加入结果集

  • 回溯: 弹出当前组合的最后一个数,以便探索该位置放其他数的可能

代码如下:

class Solution
{
private:vector<int> path;vector<vector<int>> res;public:void backTracking(int start, int end, int maxPathSize, int targetSum, int curSum){// 递归出口(纵向遍历)if (path.size() == maxPathSize){if (curSum == targetSum)res.push_back(path);return;}// 剪枝if (targetSum - curSum < start)return;// 横向遍历for (int i = start; i <= end; i++){curSum += i;path.push_back(i);                                        // 处理backTracking(i + 1, end, maxPathSize, targetSum, curSum); // 递归curSum -= i;path.pop_back(); // 回溯}}vector<vector<int>> combinationSum3(int k, int n){backTracking(1, 9, k, n, 0);return res;}
};

17. 电话号码的字母组合

点此跳转题目链接

题目描述

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。

给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

在这里插入图片描述

示例 1:

输入:digits = "23"
输出:["ad","ae","af","bd","be","bf","cd","ce","cf"]

示例 2:

输入:digits = ""
输出:[]

示例 3:

输入:digits = "2"
输出:["a","b","c"]

提示:

  • 0 <= digits.length <= 4
  • digits[i] 是范围 ['2', '9'] 的一个数字。

题解

参考:代码随想录-17

回溯算法解决。首先想清楚回溯的树形结构图:

在这里插入图片描述

然后走回溯三部曲

  • 处理: 从当前处理的数字对应的字母串中,取一个字母加入当前组合(字符串) path

  • 递归: 递归地尝试将后面的数字对应的可能字母加入组合;递归出口: path 的长度与所给数字串 digits 相同,则将 path 加入结果集

  • 回溯: 弹出当前组合的最后一个字符,以便探索该位置放其他字母的可能

c++代码如下:

class Solution
{
private:string path = "";vector<string> res;unordered_map<char, string> dict = {{'2', "abc"},{'3', "def"},{'4', "ghi"},{'5', "jkl"},{'6', "mno"},{'7', "pqrs"},{'8', "tuv"},{'9', "wxyz"}};public:void backTracking(const string &digits, int start){// 递归出口if (path.length() == digits.length()){res.push_back(path);return;}char digit = digits[start];   // 当前要处理的数字string letters = dict[digit]; // 当前处理数字对应的字母// 横向遍历for (int i = 0; i < letters.length(); i++){path.push_back(letters[i]);      // 处理backTracking(digits, start + 1); // 递归path.pop_back();                 // 回溯}}vector<string> letterCombinations(string digits){if (digits == "")return res;backTracking(digits, 0);return res;}
};

顺便再熟悉下golang:

type Helper struct {path stringres  []stringdict map[byte]string
}func newHelper() *Helper {return &Helper{dict: map[byte]string{'2': "abc",'3': "def",'4': "ghi",'5': "jkl",'6': "mno",'7': "pqrs",'8': "tuv",'9': "wxyz",},}
}func (helper *Helper) backTracking(digits string, start int) {// 递归出口if len(helper.path) == len(digits) {helper.res = append(helper.res, helper.path)return}letters := helper.dict[digits[start]]for _, letter := range letters {helper.path = helper.path + string(letter)     // 处理helper.backTracking(digits, start+1)           // 递归helper.path = helper.path[:len(helper.path)-1] // 回溯}
}func letterCombinations(digits string) []string {helper := newHelper()if len(digits) == 0 {return helper.res}helper.backTracking(digits, 0)return helper.res
}

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

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

相关文章

面向切面编程(AOP)

通知类型 Grep Console插件可右键选中日志高亮显示 正常情况 异常情况(around after和目标方法在一起&#xff0c;目标方法异常后&#xff0c;around after不执行) 通知顺序 execution 需要匹配两个没有任意交集的方法时&#xff0c;可以使用两个execution annotation 自定义…

【计算机网络】期末实验答辩

注意事项&#xff1a; 1&#xff09;每位同学要在下面做过的实验列表中选取三个实验进行答辩准备&#xff0c;并将自己的姓名&#xff0c;学号以及三个实验序号填入共享文档"1&#xff08;2&#xff09;班答辩名单"中。 2&#xff09;在答辩当日每位同学由老师在表…

支持向量机 及其分类案例详解(附Python 代码)

支持向量机分类器预测收入等级 我们将构建一个支持向量机&#xff08;SVM&#xff09;分类器&#xff0c;以预测一个人基于14个属性的收入等级。我们的目标是判断收入是否高于或低于每年$50,000。因此&#xff0c;这是一个二元分类问题。我们将使用在此处可用的人口普查收入数…

Python高维度大型气象矩阵存储策略分享

零、前情提要 最近需要分析全球范围多变量的数值预报数据&#xff0c;将grb格式的数据下载下来经过一通处理后需要将预处理数据先保存一遍&#xff0c;方便后续操作&#xff0c;处理完发现此时的数据维度很多&#xff0c;数据量巨大&#xff0c;使用不同的保存策略的解析难度和…

nowcoder bc49判断两个数的大小关系

描述 KiKi想知道从键盘输入的两个数的大小关系&#xff0c;请编程实现。 输入描述&#xff1a; 题目有多组输入数据&#xff0c;每一行输入两个整数&#xff08;范围-231~231-1&#xff09;&#xff0c;用空格分隔。 输出描述&#xff1a; 针对每行输入&#xff0c;输出两…

zotfile基础配置详解

zotfile可以将自动移动pdf到指定文件夹中&#xff0c;那么应该如何配置呢&#xff1f; 遵循极简原则&#xff0c;只需配置两个地方即可。 一、路径配置 第一处是pdf附件存放的位置&#xff0c;可以指定自己想要的地方&#xff0c;我放在了C盘的文档文件夹下。 第二处是分类法…

由恶劣事件: CrowdStrike发布案例更新导致微软全球蓝屏事件的启示

前言 网络安全公司 CrowdStrike 周四发布软件更新后&#xff0c;机场、银行、证券交易所、911 服务、交通系统、酒店、新闻媒体、医院、紧急服务等开始出现臭名昭著的蓝屏死机 (BSOD)。在看似多年来最严重的 IT 中断中&#xff0c;大规模的网络安全软件问题正在全球范围内造成…

【七】Hadoop3.3.4基于ubuntu24的分布式集群安装

文章目录 1. 下载和准备工作1.1 安装包下载1.2 前提条件 2. 安装过程STEP 1: 解压并配置Hadoop选择环境变量添加位置的原则检查环境变量是否生效 STEP 2: 配置Hadoop2.1. 修改core-site.xml2.2. 修改hdfs-site.xml2.3. 修改mapred-site.xml2.4. 修改yarn-site.xml2.5. 修改hado…

C/C++大雪纷飞代码

目录 写在前面 C语言简介 EasyX简介 大雪纷飞 运行结果 写在后面 写在前面 本期博主给大家带来了C/C实现的大雪纷飞代码&#xff0c;一起来看看吧&#xff01; 系列推荐 序号目录直达链接1爱心代码https://want595.blog.csdn.net/article/details/1363606842李峋同款跳…

Linux笔记 --- 程序入门

‘\n’换行符 通常来讲我们都是使用这个符号来进行换行的操作&#xff0c;但是这个符号不仅仅是用于换行 当标准输出文件中默认使用缓冲 &#xff0c;也就是当遇到 \n 的时候会进行刷新缓冲区&#xff08;把数据输 出&#xff09; 当打印语句后面没有换行符时 &#xff0c; 需要…

强烈推荐这 3 款让你用一次就爱上,永不想删除的软件

IcecreamPDFEditor IcecreamPDFEditor是一款功能强大的PDF编辑工具&#xff0c;具备多种编辑和查看PDF文件的功能。这款软件不仅可以方便地阅读和查看各种PDF文件&#xff0c;还可以进行编辑操作。它拥有编辑文本、注释添加、页面管理以及PDF文件保护等功能。 用户可以通过下载…

JS逆向高级爬虫

JS逆向高级爬虫 JS逆向的目的是通过运行本地JS的文件或者代码,以实现脱离他的网站和浏览器,并且还能拿到和浏览器加密一样的效果。 10.1、编码算法 【1】摘要算法&#xff1a;一切从MD5开始 MD5是一个非常常见的摘要(hash)逻辑. 其特点就是小巧. 速度快. 极难被破解. 所以,…

ELK安装(Elasticsearch+Logstash+Kibana+Filebeat)

一、简介 1.1、软件简介 ELK其实是Elasticsearch&#xff0c;Logstash 和 Kibana三个产品的首字母缩写&#xff0c;这三款都是开源产品。 1.1.1、Elasticsearch简介 Elasticsearch 是一个分布式、高扩展、高实时的搜索与数据分析引擎。它能很方便的使大量数据具有搜索、分析…

研究生选择学习Android开发的利与弊?

在开始前刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「Android的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01;产品经理可以学学Axure快…

简单的CSS样式

样式分为三种 内部样式&#xff1a;写在html文件里的样式叫内部样式 内联样式&#xff1a;写在需要的标签中 外部样式&#xff1a;写在外部css文件里 可以通过不同的选择器来选择设置指定组件的样式&#xff1a; <style>/* 写在html文件里的样式叫内部样式 *//* 选择器 *…

6.3 面向对象技术-设计模式

设计模式 创建型模式 结构型模式

C++ unordered_map

1. unordered系列关联式容器 在C98 中&#xff0c; STL 提供了底层为红黑树结构的一系列关联式容器&#xff0c;在查询时效率可达到 &#xff0c;即最差情况下需要比较红黑树的高度次&#xff0c;当树中的节点非常多时&#xff0c;查询效率也不理想。最好的查询是&#xff0c…

我在百科荣创企业实践——简易函数信号发生器(6)

对于高职教师来说,必不可少的一个任务就是参加企业实践。这个暑假,本人也没闲着,报名参加了上海市电子信息类教师企业实践。7月8日到13日,有幸来到美丽的泉城济南,远离了上海的酷暑,走进了百科荣创科技发展有限公司。在这短短的一周时间里,我结合自己的教学经验和企业的…

Leetcode 剑指 Offer II 088.使用最小花费爬楼梯

题目难度: 简单 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 数组的每个下标作为一个阶梯&#xff0c;第 i 个阶梯对应着一个非…

5 Java的基本程序设计结构(基本语法4)- 集合之ArryList 和什么是包装类?

文章目录 前言一、集合二、基本数据类型的包装类三、ArryList1 ArryList对象的创建2 ArryList常见成员方法(1)boolean add(E e) : 添加元素,返回值表示是否添加成功(2)void add(int index, E e) :在指定索引位置插入元素。(3)boolean remove(E e) : 删除第一个指定元素…