日常刷题之77-组合

题目

给定两个整数 nk,返回范围 [1, n] 中所有可能的 k 个数的组合。
你可以按 任何顺序 返回答案
提示:假设 n=5,k=3 就是需要组合出来,长度=3且内容数据是在[1,n]这个区间内的所有可能得组合
同时一个组合里面内个数字只能出现一次,[1,2,3]、[3,2,1]、[2,1,3]...等,这种只要里面的内容相同,尽管顺序不否相同都认为是同一个组合


题解

说实话,刚开始看到这个题,不知道怎么下手,看了看官方的题解,大概理解了思路,然后根据刚才理解的思路和自己的感觉慢慢摸索出最终的结果。
如果还有看了还不理解的同学,可以先用官方的代码打印出来一个结果,看是最终是要一个什么结果

比如我刚开始也看不明吧这个题目想干嘛,然后我尝试用官方的代码打印出来 n=5 k=3 的结果
[[1 2 3] [1 2 4] [1 2 5] [1 3 4] [1 3 5] [1 4 5] [2 3 4] [2 3 5] [2 4 5] [3 4 5]]

然后我就明白了,原来如此

开始动手写思路-思路1

虽然我已经看过了,知道要用递归,但是我在自己写的时候还是将复杂的东西拆分出来,拆成自己可以理解的,容易看懂且方便继续往下想的思路

func combine1(n int, k int) [][]int {// 根据题目可以确定 k 不可能大于 nmResult := make([][]int, 0)mSlice := make([]int, k)mIdx := 0// 判断当前的组合长度是否满足// 秉承一个条件,就是确定第一位是那个数字,后面的数字不能比第一个数字小// 然后就是枚举出所有可能的下一位数字的排序,一直重复这个想法// 当前的第一个数字确定mFirstNum := 1// 将这个数插入数组中mSlice[mIdx] = mFirstNum// 判断这个数组插入的数据是否已经满足足够的数量if mIdx == k-1 {// 满足,则将这个数组插入到结果集中mTempSlice := make([]int, k)copy(mTempSlice, mSlice)mResult = append(mResult, mTempSlice)}// 满足的后面就不用插入了,考虑不使用插入的最后这个数据,看是否可以继续替换别的数据继续进行return mResult
}
写到这里看是否合理-思路2

上面的代码片段写了后想了想,好像是可行的,既然是可行的,继续完善,当然这里我是知道最后是要用递归写的,所以后续写的思路会偏向递归的想法

func combine2(n int, k int) [][]int {mResult := make([][]int, 0)mSlice := make([]int, k)mIdx := 0// ==============================================// 当前使用到的数据mCurNum := 1if mCurNum > n {// 当前使用的数字已经超了规定,直接退出就行}if mIdx >= k {// 已经超过数组的长度了,后面的都不用算了,直接可以退出了}// 后续这里还可以提前预判一波,判断后面剩余的长度能否塞满数组,不能的话也没必要继续下去了// 这里直接将这个数插入到数组中mSlice[mIdx] = mCurNummIdx++mCurNum++// 判断当前的数组是否刚好塞满,如果刚好塞满就可以将其插入到返回数组中if mIdx == k {mTempSlice := make([]int, k)copy(mTempSlice, mSlice)mResult = append(mResult, mTempSlice)} else {// 是如果没有塞满,继续刚才的步骤,判断数字大小、数组长度、预判后续长度是否满足 ...}// 到这里就感觉好像中间部分的数据组合还有遗漏的,上面的部分只考虑到了替换 Slice[k] 这个位置数据,// 前面部分都是固定的,前面可能还有很多种组合没有实现// 这里就要考虑一下前面部分的,比如现在是第一个数,就不要 1 这个数字插入数组mIdx--// 前面选择的数字已经自增了,这里将数组的下表调回去,然后进行上面同样的判断计算// 先 判断数字大小、数组长度、预判后续长度是否满足 ...// ==============================================return mResult
}
答案的雏形-思路3

到这里代码应该怎么写的雏形已经出来了,然后以及哪些参数是需要进行传递的,心里大致都会有数了,然后进行代码填充

func execFunc(aResult *[][]int, aSlice []int, aIdx, aCurNum, n int) {if aCurNum > n {return}if aIdx >= len(aSlice) {return}// n-aCurNum+1 假设最大值n=5 当前插入数字aCurNum=4 此时5-4+1=2 表示还有2个数(4、5)可以使用// 然后加上数组已经插入的个数 aIdx 如果还不够填充数组那么后面的都不会满足条件if n-aCurNum+1+aIdx < len(aSlice) {return}aSlice[aIdx] = aCurNumif aIdx+1 == len(aSlice) {mTempSlice := make([]int, len(aSlice))copy(mTempSlice, aSlice)*aResult = append(*aResult, mTempSlice)}// 如果数据还没填满,继续进行(数据填满了也会执行到这里,会在里面的判断直接退出了)execFunc(aResult, aSlice, aIdx+1, aCurNum+1, n)// 这里就考虑前面部分,假设这里没有将这个值插入到数组中,从而进行下个值的插入execFunc(aResult, aSlice, aIdx, aCurNum+1, n)
}func combine3(n int, k int) [][]int {mResult := make([][]int, 0)mSlice := make([]int, k)execFunc(&mResult, mSlice, 0, 1, n)return mResult
}

到这里可以说是结束了,但是这递归方法的参数有点多,并且提交的运行结果内存占用并不是很满意,我想着官方的代码中用到了闭包,我感觉那种好像会省点内存,毕竟在go中方法的入参大部分都是值传递(可以理解为把传进来的参数复制了一份使用的)

另一种写法

在思路完全不变的情况下,使用上官方的套路看看,结果就是确实有点用

func combine4(n int, k int) [][]int {mResult := make([][]int, 0)mSlice := make([]int, k)var mFunc func(aIdx, aCurNum int)mFunc = func(aIdx, aCurNum int) {if aCurNum > n {return}if aIdx >= k {return}if n-aCurNum+1+aIdx < k {return}mSlice[aIdx] = aCurNumif aIdx+1 == k {mTempSlice := make([]int, k)copy(mTempSlice, mSlice)mResult = append(mResult, mTempSlice)}mFunc(aIdx+1, aCurNum+1)mFunc(aIdx, aCurNum+1)}mFunc(0, 1)return mResult
}

提交记录

在这里插入图片描述
在这里插入图片描述


题目来源:力扣题库


一点点笔记,以便以后翻阅。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

日增500粉的全自动引流神器

全自动引流&#xff0c;采集曝光一体每天截流500&#xff0c;这是每个企业主和网红的终极梦想。今日&#xff0c;我将分享一套高效而可行的引流策略&#xff0c;帮助你实现这个目标。 我们需要理解&#xff0c;全自动引流并不意味着无人参与。相反&#xff0c;它仍需要你的参与…

【王道训练营】第6题 输入一个整型数,判断是否是对称数,如果是,输出yes,否则输出no

文章目录 我的代码改正代码其他代码 我的代码 没有完成 #include<stdio.h> int main(){int a;int b;int c0;//位数int d0;//比较几次scanf("%d",&a);while(b!0){bb/10;c;}dc/2;//比较几次int ffor(int i0 ;i<d;i){int ec;//位数fa - a / (((e-i-1)*10…

算法笔记~—位运算

目录 常见位运算&#xff1a; 1、基础位运算 2、对于一个数n。确定、修改这个数n二进制x位。 3、提取&#xff08;确定&#xff09;一个数n最右侧的1&#xff08;bit&#xff09;与干掉最右侧的1&#xff08;bit&#xff09; 4、异或运算律 5、位运算的优先级&#xff1a…

Go --- Go语言垃圾处理

概念 垃圾回收&#xff08;GC-Garbage Collection&#xff09;暂停程序业务逻辑SWT&#xff08;stop the world&#xff09;程序根节点&#xff1a;程序中被直接或间接引用的对象集合&#xff0c;能通过他们找出所有可以被访问到的对象&#xff0c;所以Go程序的根节点通常包括…

虚拟机Linux-openEuler硬盘空间扩容

虚拟机Linux-openEuler硬盘空间扩容 1、需求场景 我们在使用虚拟机时&#xff0c;可能会出现磁盘空间不够用导致各种bug出现的情况。 首先&#xff0c;我们要扩展虚拟机的可用磁盘空间。如图所示&#xff0c;我的原本硬盘大小为8G&#xff0c;我们扩展到30GB 2、打开虚拟机…

【git分支管理策略】如何高效的管理好代码版本

目录 1.分支管理策略 2.我用的分支管理策略 3.一些常见问题 1.分支管理策略 分支管理策略就是一些经过实践后总结出来的可靠的分支管理的办法&#xff0c;让分支之间能科学合理、高效的进行协作&#xff0c;帮助我们在整个开发流程中合理的管理好代码版本。 目前有两套Git…

【GameFramework框架内置模块】16、配置(Setting)

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址QQ群&#xff1a;398291828 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 【GameFramework框架】系列教程目录&#xff1a;…

2024年软件测试,“我“从初级到高级进阶,不再走弯路...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 现在2024年&#…

54、Qt/对话框、事件机制相关学习20240325

一、完善对话框&#xff0c;点击登录按钮&#xff0c;如果账号和密码匹配&#xff0c;则弹出信息对话框&#xff0c;给出提示”登录成功“&#xff0c;提供一个Ok按钮&#xff0c;用户点击Ok后&#xff0c;关闭登录界面&#xff0c;跳转到其他界面 如果账号和密码不匹配&#…

python(django)之单一接口管理功能后台开发

1、创建数据模型 在apitest/models.py下加入以下代码 class Apis(models.Model):Product models.ForeignKey(product.Product, on_deletemodels.CASCADE, nullTrue)# 关联产品IDapiname models.CharField(接口名称, max_length100)apiurl models.CharField(接口地址, max_…

服务运营|香港大学雷骁:收益管理中价格歧视的公平性

编者按&#xff1a; INFORMS George B. Dantzig Dissertation Award 用于表彰运筹学和管理科学领域中具有创新性和实用性的最佳毕业设计。香港大学助理教授雷骁题为“Revenue Management in Video Games and With Fairness” 是这一奖项2023年度的提名者之一。 这篇毕业设计重…

PMSM 永磁同步电机滑膜控制 SVPWM矢量控制 matlab simulink 仿真

仿真搭建平台&#xff1a; (1)该模型采用matlab/simulink 2016b版本搭建&#xff0c;使用matlab 2016b及以上版本打开最佳; (2)该模型已经提前转换了各个常用版本&#xff08;最低为matlab2012b&#xff09;&#xff0c;防止出现提示版本过高的情况。 模型截图&#xff1a; 算…

网游版五子棋

五子棋游戏属于开房间类休闲游戏。可以非常方便实现分布式战斗服横向拓展&#xff0c;只要感觉服务器有压力&#xff0c;可以通过动态加战斗服服务器来实现。本文介绍一个基于jforgame组件开发的五子棋网络小游戏&#xff0c;支持分布式部署战斗服。 1.通信组件 浏览器&#…

Linux:进程概念认识

进程 基本概念 课本概念&#xff1a;程序的一个执行实例&#xff0c;正在执行的程序等 内核观点&#xff1a;担当分配系统资源&#xff08; CPU 时间&#xff0c;内存&#xff09;的实体。 描述进程 -PCB 进程信息被放在一个叫做进程控制块的数据结构中&#xff0c;可以理解为…

34.基于SpringBoot + Vue实现的前后端分离-足球俱乐部管理系统(项目 + 论文)

项目介绍 系统包含用户、教练、管理员三个角色 用户&#xff1a;登录、注册、查看俱乐部公告信息、查看俱乐部赛事信息、个人中心等教练&#xff1a;登录、个人中心、用户管理、赛事管理、球员数据管理、训练计划管理、公告信息管理等管理员&#xff1a;登录、个人中心、教练…

Harmony OS 网络编程 实验指南

netcat简介 netcat 是什么&#xff1f; netcat是一个非常强大的网络实用工具&#xff0c;可以用它来调试TCP/UDP应用程序&#xff1b; netcat 如何安装&#xff1f; Linux上可以使用发行版的包管理器安装&#xff0c;例如Debian/Ubuntu上&#xff1a; sudo apt-get instal…

day07-缓存商品、购物车

1. 缓存菜品 1.1 问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得&#xff0c;如果用户端访问量比较大&#xff0c;数据库访问压力随之增大。 结果&#xff1a; 系统响应慢、用户体验差 1.2 实现思路 通过Redis来缓存菜品数据&#xff0c;减少数据库查询操作。 缓…

Git_.gitignore文件相关知识

.gitignore 作用&#xff1a;指明不对哪些文件进行版本控制。 应当忽略哪些文件&#xff1f; 系统或软件自动生成的文件编译时产生的中间文件和结果文件运行时产生的日志文件&#xff0c;临时文件和缓存文件涉及身份&#xff0c;密码&#xff0c;口令&#xff0c;秘钥等敏感…

Java——基于CompletableFuture的流水线并行处理

CompletableFuture在JDK1.8提供了一种更加强大的异步编程的api。它实现了Future接口&#xff0c;也就是Future的功能特性CompletableFuture也有&#xff1b;除此之外&#xff0c;它也实现了CompletionStage接口&#xff0c;CompletionStage接口定义了任务编排的方法&#xff0c…

C语言结构体之位段

位段&#xff08;节约内存&#xff09;&#xff0c;和王者段位联想记忆 位段是为了节约内存的。刚好和结构体相反。 那么什么是位段呢&#xff1f;我们现引入情景&#xff1a;我么如果要记录一个人是男是女&#xff0c;用数字0 1表示。我们发现只要一个bit内存就可以完成我们想…