代码随想录算法训练营第二十四天| 理论基础,77. 组合

 题目与题解

参考资料:回溯法理论基础

带你学透回溯算法(理论篇)| 回溯法精讲!_哔哩哔哩_bilibili

77. 组合 

题目链接:​​​​​​​​​​​​​​77. 组合 

代码随想录题解:77. 组合 

视频讲解:带你学透回溯算法-组合问题(对应力扣题目:77.组合)| 回溯法精讲!_哔哩哔哩_bilibili

带你学透回溯算法-组合问题的剪枝操作(对应力扣题目:77.组合)| 回溯法精讲!_哔哩哔哩_bilibili

解题思路:

        回溯法的题目之前很少做,对于这一道题,手写穷举是很好写的,如果只有两个数的组合,求两层for循环即可,但是当k很大时,就比较难直接暴力写for循环了。

        这一题主要是初步体验一下回溯法的流程,所以直接看答案了。

看完代码随想录之后的想法 

        回溯法的要点:

  1. 回溯法解决的问题都可以抽象为树形结构(N叉树),树的每深入一层相当于递归操作,而在每层上的操作一般都是循环处理。
  2. 每次搜索到了叶子节点,我们就找到了一个结果,因此找到叶子节点一般是终止条件。
  3. 为了提高穷举效率,可以针对N叉树做剪枝操作,避免不必要的递归和循环。

回溯法一般的函数体为:

void backtracking(参数) {if (终止条件) {存放结果;return;}for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {处理节点;backtracking(路径,选择列表); // 递归回溯,撤销处理结果}
}

对于这道题,其N叉树为

 

每次循环的起始值startIndex就是取1,2,3...的操作,而终止条件就是当前循环的一个符合条件的list已经满了,达到了叶子节点,将结果塞入result的list后,还需要将list最末尾的节点弹出,方便下一次操作。

class Solution {List<List<Integer>> result= new ArrayList<>();LinkedList<Integer> path = new LinkedList<>();public List<List<Integer>> combine(int n, int k) {backtracking(n,k,1);return result;}public void backtracking(int n,int k,int startIndex){if (path.size() == k){result.add(new ArrayList<>(path));return;}for (int i =startIndex;i<=n;i++){path.add(i);backtracking(n,k,i+1);path.removeLast();}}
}

当然,考虑到如果从startIndex到n的元素数目不足k,那么就算循环结束这个path也不会被使用,浪费了时间,可以提前剪枝,因此循环中i的上限设置为n - (k - path.size()) + 1 ,注意path.size()是小于等于k的。这样就完成了剪枝。

class Solution {List<List<Integer>> result = new ArrayList<>();LinkedList<Integer> path = new LinkedList<>();public List<List<Integer>> combine(int n, int k) {combineHelper(n, k, 1);return result;}/*** 每次从集合中选取元素,可选择的范围随着选择的进行而收缩,调整可选择的范围,就是要靠startIndex* @param startIndex 用来记录本层递归的中,集合从哪里开始遍历(集合就是[1,...,n] )。*/private void combineHelper(int n, int k, int startIndex){//终止条件if (path.size() == k){result.add(new ArrayList<>(path));return;}for (int i = startIndex; i <= n - (k - path.size()) + 1; i++){path.add(i);combineHelper(n, k, i + 1);path.removeLast();}}
}

遇到的困难

        我一开始按照答案的思路抄写,但是没有用linkedlist作为存储path的变量而是用了list,在result.add的时候直接插入了path,结果每次输出的结果都是全空的。debug的时候很奇怪,插入result的一瞬间结果是对的,到下一层遍历的时候result里面的元素竟然减少了。就很奇怪。

        查阅了以后才知道,java中的list如果直接被加入list.add(list),传入的是list的引用,而非拷贝。所以即使当时把path加入了result,随着后面回溯时要弹出元素,被引用加入result的path自然也会随之改变。

        正确的做法是每次拷贝一份新的path,即不用result.add(path)而是result.add(new ArrayList<>(path)),这样path就算有任何变化,也不会影响这一份拷贝。

今日收获

        初步学习了一下回溯法的思路,并且用组合这道题为例理解了一下回溯法,还学习了java中list的实际传递方式。不过还是有点云里雾里的,需要再做一点题来熟悉。

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

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

相关文章

InputStreamReader类详解

咦咦咦&#xff0c;各位小可爱&#xff0c;我是你们的好伙伴——bug菌&#xff0c;今天又来给大家普及Java SE相关知识点了&#xff0c;别躲起来啊&#xff0c;听我讲干货还不快点赞&#xff0c;赞多了我就有动力讲得更嗨啦&#xff01;所以呀&#xff0c;养成先点赞后阅读的好…

【常见面试题】Golang中,协程数最多可以开多少个?

参考&#xff1a; Goroutine 究竟可以开多少&#xff1f; 一、先说结论&#xff1a; 能开多少个协程&#xff0c;取决于单个协程处理方法所占用的CPU和内存资源&#xff08;也就是看你计算机运行的应用程序的具体代码逻辑&#xff09;。 二、具体来说&#xff1a; 如果是C…

深入解析Windows Edge兼容性问题及其修复策略

深入解析Windows Edge兼容性问题及其修复策略 目录 深入解析Windows Edge兼容性问题及其修复策略 一、引言 二、Windows Edge浏览器概述 三、深入分析兼容性问题 四、案例研究 五、技术解决方案与最佳实践 六、数据图表与性能指标 七、编程技巧与代码示例 一、引言 随着W…

Radash一款JavaScript最新的实用工具库,Lodash的平替!

文章目录 Lodash 的痛点进入正题--Radash特点 举例几个常用的api 一说lodash应该大部分前端同学都知道吧&#xff0c;陪伴我们好多年的JavaScript工具库&#xff0c;但是自从 ES6 出现后就慢慢退出前端人的视线&#xff0c;能ES6写的代码绝对不会用Lodash&#xff0c;也不是完全…

基于STC12C5A60S2系列1T 8051单片机的一个按键长按开关机后一个按键单击长按都增加数值另一个按键单击长按都减少数值应用

基于STC12C5A60S2系列1T 8051单片机的一个按键长按开关机后一个按键单击长按都增加数值另一个按键单击长按都减少数值应用 STC12C5A60S2系列1T 8051单片机管脚图STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介…

深入理解Java接口:定义、使用与重要性(day13)

导语&#xff1a;Java接口是Java编程语言中的一个核心概念&#xff0c;它提供了一种定义方法但不包含方法实现的方式。接口在Java编程中扮演着重要角色&#xff0c;能够帮助我们实现代码的高内聚、低耦合&#xff0c;提高代码的复用性和可维护性。本文将详细介绍Java接口的定义…

SQL复习专题

请结合B站-技术蛋老师 视频学习 核心语法 一、增&#xff1a;数据库/表格 create create database 数据库名&#xff1b;#创建表&#xff08;列名类型&#xff09; mysql> create table eggs_record(-> id int,-> egg_name varchar(10),-> sold date-> ); 这…

Android熄屏/亮屏,旋转屏幕/横竖屏切换生命周期变化与activity销毁重建

Android熄屏/亮屏&#xff0c;旋转屏幕/横竖屏切换生命周期变化与activity销毁重建 1、熄屏/亮屏 熄屏后&#xff0c;Android生命周期走&#xff1a; onPause onStop 接着点亮Android手机屏幕&#xff0c;生命周期走&#xff1a; onRestart onStart onResume 2、旋转屏幕&…

服务器中有g++,但是查询不到,Command ‘g++‘ not found

有gcc但是查询不到g&#xff0c;gcc版本为9.5.0 (base) zyICML:~$ g -V Command g not found, but can be installed with: apt install g Please ask your administrator. 突然就出现这个问题&#xff0c;导致detectron装不上&#xff0c;现在有时间了专门研究下怎么解决 这…

阿里云8核32G服务器g8y租用优惠价格243.30元/月

阿里云8核32G服务器g8y租用优惠价格243.30元/月&#xff0c;配置云服务器g8y&#xff08;ecs.g8y.2xlarge&#xff09;8核32G、按固定带宽1M-5M、可选ESSD云盘40G起&#xff0c;优惠价格243.30元一个月&#xff0c; 阿里云服务器优惠活动 aliyunfuwuqi.com/go/aliyun 阿里云服务…

【重制版】在Android手机上安装kali Linux

前言 由于kali官方的Nethunter2的安装代码因为…无法访问&#xff0c;手头又没有一些受支持的机器3&#xff0c;所以做了这个脚本&#xff0c;供大家使用。 工具 搭载基于Android的手机TermuxVNC Viewer 安装必备软件(如已安装请忽略) 请到 https://www.hestudio.net/post…

Spring中 Bean生命周期总结

Bean生命周期从创建到销毁经历多个阶段&#xff0c;以下是这一过程的主要步骤&#xff1a; 1. 实例化Bean 首先&#xff0c;Spring容器通过构造器&#xff08;或者在某些情况下是通过工厂方法&#xff09;创建Bean的实例。 Bean的实例化主要可以通过以下三种方式进行&#x…

公安可视化分析指挥调度平台的构成要素

匠心接单中...8年前端开发和UI设计接单经验&#xff0c;完工项目1000&#xff0c;持续为友友们分享有价值、有见地的干货观点&#xff0c;有业务需求的老铁&#xff0c;欢迎关注发私信。 公安可视化分析指挥调度平台是一个集成了多种技术和功能的综合性平台&#xff0c;主要由…

面试宝典:深入分析golang 的反射(reflect)

Go 语言&#xff08;Golang&#xff09;是一门静态强类型、编译型的编程语言&#xff0c;由 Google 开发。Go 语言的反射&#xff08;Reflection&#xff09;是一个强大的特性&#xff0c;它允许程序在运行时获取类型的信息&#xff0c;并且能够操作这些类型。反射在 Go 语言中…

DataLoader;model_best.eval():设置模型为评估模式:

目录 DataLoader model_best.eval():设置模型为评估模式: DataLoader 您正在使用PaddlePaddle框架的DataLoader来创建一个测试数据加载器。这个加载器会从FoodDataset数据集中读取数据,并且按照指定的参数进行配置。以下是对您提供的代码片段的详细解释: pythonbatch_siz…

mysql 日期时间函数

学习了字符串函数&#xff0c;接着学习日期时间函数&#xff0c; 3&#xff0c;日期和时间函数 日期和时间函数主要用来处理日期和时间值&#xff0c;一般的日期函数除了使用DATE类型的参数外&#xff0c;也可以使用DATETIME或者TIMESTAMP类型的参数&#xff0c;但会忽略这些值…

Discuz采集发布插件

Discuz&#xff08;简称DZ&#xff09;是一款知名的开源论坛系统&#xff0c;广泛应用于各类网站社区。对于许多站长来说&#xff0c;保持论坛内容的更新是一项挑战&#xff0c;特别是在内容量庞大的情况下。为了解决这个问题&#xff0c;有一类特殊的插件是用于在Discuz论坛中…

康耐视visionpro-CogDistancePointLineTool工具详细说明

CogDistancePointLineTool功能说明: 测量点到线的距离 备注&#xff1a;在“Geometry-Measurement”选项中的所有工具都是测量尺寸或角度工具&#xff0c;包括测量线与线 的角度、点与线的距离、圆与圆的距离等测量工具&#xff0c;工具使用的方法相似。 ①.打开工具栏&#…

C++开源项目研究——gh0st远控(一)

上一节我们讲过肉机最关键的一步就是通过connect来连接指定的主控端 if (connect(m_Socket, (SOCKADDR *)&ClientAddr, sizeof(ClientAddr)) SOCKET_ERROR) return false; 其实在次之前应当是主控端先监听相应的端口&#xff0c;然后肉机再来连接这个端口的 在主控端…

分寝室(20分)(JAVA)

目录 题目描述 输入格式&#xff1a; 输出格式&#xff1a; 输入样例 1&#xff1a; 输出样例 1&#xff1a; 输入样例 2&#xff1a; 输出样例 2&#xff1a; 题解&#xff1a; 题目描述 学校新建了宿舍楼&#xff0c;共有 n 间寝室。等待分配的学生中&#xff0c;有女…