力扣第77题 组合 c++ 回溯经典题 注释加优化 代码

news2025/1/12 1:03:56

题目

77. 组合

中等

相关标签

回溯

给定两个整数 n 和 k,返回范围 [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

思路和解题方法

在类 Solution 中定义了两个成员变量:

  • ans:用于存储所有组合的二维数组。
  • vec:用于存储当前组合的一维数组。

        接下来,有一个私有方法 backtrackint(),该方法使用递归和回溯的方式生成所有的组合。方法的参数包括 n(总范围)、k(选择数量)和 startIndex(当前选择的数字范围的起始位置)。

        在 backtrackint() 方法中,首先进行终止条件的判断。如果当前组合的大小等于 k,说明已经选取了足够数量的数字,将当前组合加入到结果数组 ans 中,并返回。

然后,使用一个循环从 startIndexn 进行遍历。在每次循环中,将当前的数字加入到当前组合中(即 vec.push_back(i)),然后通过递归调用 backtrackint() 函数,在从 i+1n 的范围内选择下一个数字。递归调用完成后,需要进行回溯操作,即将上一步加入的数字移出当前组合(即 vec.pop_back()),以便继续生成其他组合。

        在主函数 combine() 中,首先清空存储结果的数组 ans 和当前组合的数组 vec。然后调用 backtrackint() 方法开始生成由 1~n 中选取 k 个数字的所有组合。最后返回结果数组 ans

        这段代码的核心思想是利用回溯算法,在搜索过程中逐步构建组合,并在选取完 k 个数字后将组合加入到结果数组中,同时进行回溯操作以继续生成其他组合。

复杂度

        时间复杂度:

                O(C(n,k))

时间复杂度是 O(C(n,k)),其中 n 表示数字范围的大小,k 表示选择的数字数量。C(n, k) 是组合数,表示从 n 个数字中选取 k 个数字的组合数量。

        空间复杂度

                O(k)

空间复杂度是 O(k),其中 k 表示选择的数字数量。主要的空间消耗在结果数组 ans 和当前组合数组 vec 上。递归调用的深度为 k,每次递归调用都需要创建一个新的组合数组 vec,并且结果数组 ans 也会存储最终的组合结果。

c++ 代码

class Solution {
public:
    vector<vector<int>> ans; // 存储所有组合的二维数组
    vector<int> vec; // 存储当前组合的一维数组

    // 递归回溯函数,startIndex 表示当前选择的数字范围的起始位置
    void backtrackint(int n, int k, int startIndex) {
        if(vec.size() == k) { // 如果已经选了 k 个数字,将当前组合加入结果数组
            ans.push_back(vec);
            return;
        }

        for(int i = startIndex; i <= n; i++) { // 在从 startIndex 到 n 中选一个数字加入当前组合
            vec.push_back(i);
            backtrackint(n, k, i + 1); // 递归,在从 i+1 到 n 中选下一个数字
            vec.pop_back(); // 回溯,撤销上一步加入的数字
        }
    }

    vector<vector<int>> combine(int n, int k) {
        ans.clear();
        vec.clear();
        backtrackint(n, k, 1);
        return ans;
    }
};

优化 具体解释

path 长度达到 k 时,将 path 中的数字加入到结果数组中,并返回。

然后通过循环枚举可选数字的起始位置,遍历从 startIndex(n - (k - path.size()) + 1) 的数字。这里需要解释一下 (n - (k - path.size()) + 1),其实就是用来保证从当前数字一直取到最后一个数字,至少还剩下 k - path.size() 个数字,否则就不可能凑够 k 个数字。

在循环内部,首先将 i 加入到 path 中,然后递归调用自己,但此时的起始位置为 i+1。等到递归调用返回时,回溯撤销已经加入路径的数字,以便继续尝试其他数字。

c++优化后的代码

class Solution {
private:
    vector<vector<int>> result; // 存储所有符合条件的组合
    vector<int> path; // 当前的组合
    void backtracking(int n, int k, int startIndex) {
        if (path.size() == k) { // 如果当前的组合大小已经等于 k,说明已经选取了足够数量的数字,将当前组合加入到结果数组中
            result.push_back(path); // 添加当前的组合到结果数组中
            return; // 返回
        }
        for (int i = startIndex; i <= n - (k - path.size()) + 1; i++) { // 重点优化处 遍历可选的节点
            path.push_back(i); // 处理节点,将当前的数字加入到当前组合中
            backtracking(n, k, i + 1); // 递归调用,从 i+1 到 n 的范围内选择下一个数字
            path.pop_back(); // 回溯,撤销处理的节点,将上一步加入的数字移出当前组合
        }
    }
public:
    vector<vector<int>> combine(int n, int k) {
        backtracking(n, k, 1); // 从 1 开始进行回溯
        return result; // 返回所有符合条件的组合
    }
};

觉得有用的话可以点点赞,支持一下。

如果愿意的话关注一下。会对你有更多的帮助。

每天都会不定时更新哦  >人<  。

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

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

相关文章

基于Java的健身俱乐部管理系统设计与实现(亮点:健身课程课程、会员下单、在线支付)

文章目录 前言具体实现截图详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09; 代码参考数据库参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&#xff0c…

怎么查苹果手机激活时间?详细查看指南来了!

苹果手机之所以受到广大消费者的喜爱&#xff0c;是因为它提供了优质的用户体验、强大的生态系统以及美观的设计等等。很多小伙伴在购买iPhone后都会担心一个问题&#xff1a;手机是否为新机&#xff0c;手机有没有被激活过&#xff1f;那么&#xff0c;怎么查苹果手机激活时间…

电压放大器在铁电材料表征中的应用有哪些

电压放大器在铁电材料表征中具有广泛的应用。铁电材料是一类特殊的功能材料&#xff0c;具有独特的电荷分布和结构变化特性&#xff0c;在电子器件、传感器、存储器等领域具有重要的应用价值。而电压放大器作为一种重要的电子器件&#xff0c;可以提供稳定可靠的信号放大功能&a…

mysql分组排序并取每组的前1条记录

实际开发中会遇到使用mysql分组统计&#xff0c;并且要求取每组数据的前1条数据 如下&#xff1a;需要根据模块分组&#xff0c;取每组中的前1条优质供应商数据&#xff0c;取前几条数据&#xff0c;其实会用到上篇文章中的排名函数row_number() 函数来实现&#xff0c;其含义…

开源原生android的视频编辑软件

videoEditAndroid 介绍 开源原生android的视频编辑软件 本人android 新手,也是边写边学习中,感觉写的很乱&#xff0c;功能虽已实现,但是会不断优化代码 也欢迎有兴趣的小伙伴加入 码农不易&#xff0c;欢迎 star 项目页面功能完成列表 视频选择(待完善) 静音 视频编辑 导…

AC修炼计划(AtCoder Regular Contest 165)

传送门&#xff1a;AtCoder Regular Contest 165 - AtCoder 本次习题参考了樱雪猫大佬的题解&#xff0c;大佬的题解传送门如下&#xff1a;Atcoder Regular Contest 165 - 樱雪喵 - 博客园 (cnblogs.com) A - Sum equals LCM 第一题不算特别难 B - Sliding Window Sort 2 对…

Latex伪代码Switch Case(简单暴力版)

自己添加新命令的Switch Case&#xff0c;用的algorithm和algorithmicx安装包 格式可调整&#xff01; 效果图&#xff1a; 代码&#xff1a; \usepackage{algorithm} \usepackage{algorithmicx} \newcommand{\SWITCH}[1]{\textbf{switch} (#1)} \newcommand{\ENDSWITCH}{\t…

【python海洋专题二十二】在海图上text

上期出了四个子图 但是想标记A,B,C,D 或者春夏秋冬 本期内容 在海图上text 1:效果如图 未标记 标记 关键语句 # 添加文本注释 ax.text(104, 22, autumn, fontdict={family: serif, size: 4, color: blue}, ha=center, va=center, transform=ccrs.PlateCarree())往期推荐…

分享一下做一个电商小程序的步骤是什么呢

在当今互联网时代&#xff0c;电商小程序已经成为了一种新的商业模式&#xff0c;它不仅能够提高用户的购物体验&#xff0c;还能为商家带来更多的销售机会。本文将详细介绍电商小程序的步骤&#xff0c;帮助读者了解如何从零开始制作一个电商小程序 首先&#xff0c;让我们来了…

日期格式转换

标准格式转换&#xff1a; 1.这里获取当前系统的时 2.这里的Date类是 java.util 3.默认输出的日期格式是国外的方式&#xff0c;因此通常需要格式转换 Date date new Date();System.out.println("当前日期&#xff1a;"date);Date date1 new Date(954545424);Syst…

阿里云“玩转云上 StarRocks3.0 湖仓分析”,开启数据分析新范式

简介&#xff1a; 阿里云 EMR OLAP 团队与 StarRocks 社区联合出品&#xff0c;玩转云上 StarRocks3.0 湖仓分析训练营&#xff0c;围绕 StarRocks3.0 系列解读、EMR Serverless StarRocks 存算分离功能与应用场景介绍&#xff0c;开启数据分析新范式&#xff01; StarRocks3…

时间复杂度为 O(n^2) 的排序算法

大家好&#xff0c;我是 方圆。对于小规模数据&#xff0c;我们可以选用时间复杂度为 O(n2) 的排序算法&#xff0c;因为时间复杂度并不代表实际代码的执行时间&#xff0c;而且它也省去了低阶、系数和常数&#xff0c;仅代表的增长趋势&#xff0c;所以在小规模数据情况下&…

PAM从入门到精通(一)

本文参考以下博文&#xff1a; PAM模块详解及sudo命令 PAM 的应用开发和内部实现源码分析 PAM详解&#xff08;一&#xff09;PAM介绍 百度百科 —— PAM Oracle Solaris 10 开发者安全性指南 —— PAM 框架介绍 特此致谢&#xff01; 零、引言 身份认证是操作系统安全的…

Oracle-truncate误删数据恢复

前言&#xff1a; truncate操作误删数据之后想恢复数据通常比较困难&#xff0c;因为truncate操作属于ddl操作无法使用直接undo闪回查询方式恢复数据&#xff0c;并且由于空间大小、备份时间以及变更操作不规范原因&#xff0c;往往在执行操作之前没有对表进行CTAS备份或者其他…

【Java】枚举 Enum

枚举 Enum 枚举的使用Enum 类的常用方法枚举优缺点用枚举实现一个单例模式 枚举的主要用途是&#xff1a;将一组常量组织起来&#xff0c;在这之前表示一组常量通常使用定义常量的方式&#xff1a; public static int final RED 1; public static int final GREEN 2; publi…

vue3 状态管理pinia

1. 什么是Pinia Pinia 是 Vue 的专属的最新状态管理库 &#xff0c;是 Vuex 状态管理工具的替代品 特点优势: 提供更加简单的API(去掉了mutation)提供符合组合式风格的API(和Vue3新语法统一)去掉modules的概念,每一个store都是一个独立的模块配合TypeScript更加友好,提供可靠的…

微信里写周报添加到公司办公平台的方法

概要 微信和公司用的办公平台互通&#xff0c;我们已经说过几篇。 这次将给大家介绍&#xff0c;如何在微信里写周报内容&#xff0c;添加到公司办公平台的周报应用里。 工作中&#xff0c;如果出差在外&#xff0c;或者回到家想要汇报一周的工作情况的时候&#xff0c;不用…

Druid连接池最小连接数设置失效问题

问题发现&#xff1a; 配置 当项目启动后 线程池确实是初始化了5条连接&#xff0c;但是当项目运行一段时间后&#xff0c;5条连接确消失了&#xff0c;只会程序用到得时候&#xff0c;再去初始化连接&#xff0c;这样有点违背了参数设置得意义&#xff0c;后来通过查阅资料发…

简单好用的文档管理系统MinDoc

什么是 MinDoc &#xff1f; MinDoc 是一款针对 IT团队开发的简单好用的文档管理系统。MinDoc 的前身是 SmartWiki 文档系统。SmartWiki 是基于 PHP 框架 laravel 开发的一款文档管理系统。因 PHP 的部署对普通用户来说太复杂&#xff0c;所以改用 Golang 开发。可以方便用户部…

vue 01

安装vscode 按照如下方式汉化 给vscode 安装插件 Vetur 安装node.js 下载地址https://nodejs.org/en/download/&#xff0c;选择windows msi 在cmd下检查如下&#xff1a; 检查nodejs版本:node --version 检查npm的安装版本&#xff1a;npm -v 执行命令&#xff1a; npm i…