热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

算法_回溯_组合总和

文章目录组合总和1.解法2.优化3.总结组合总和leetcode链接1.解法这道题和普通的组合问题的区别在于它可以重复取集合中的数值,所以我们再代码实现的时候需

文章目录

  • 组合总和
    • 1.解法
    • 2.优化
    • 3.总结


组合总和

leetcode链接

1.解法

这道题和普通的组合问题的区别在于它可以重复取集合中的数值,所以我们再代码实现的时候需要注意这一点。

因为组合问题就是回溯问题,而回溯问题都可以画成树,所以我们先画出图帮助分析:
在这里插入图片描述
从图中我们看出一个规律,就是当取得了一个元素后,我们在下一次选择数值的时候,集合就变为了该数值往后的数值集合(包括该数)。举个例子,如果取了2,那么下一次就要在[2,5,3]中选。如果取了5,下一次就要在[5,3]中取。这为我们实现逻辑部分的代码提供了依据。

然后是回溯三部曲:

  1. 确定函数头和一些辅助变量:

    result = [] # 存储最终稿答案path = [] # 存储当前取得的所有数值def backtracking(candidates,target,sum)

    因为参数是要在编写代码中确定的,所以我这里先写上这三个参数,后面看完代码大家就懂了。但这里还是简单说一下含义,

    candidates代表的是当前取数的集合,
    target代表的是目标数值,
    sum代表的是已取得的数的总和。

  2. 确定递归出口:

    if sum > target:returnif sum == target:result.append(path)return

    (1)当已取得的数值的和,即sum,比目标数值大的时候,就直接退出该层递归;

    (2)当已取得的数值的和,即sum,和目标数值一样大的时候,就可以把取得的数值保存一下,然后退出该层递归。

  3. 确定递归逻辑:

    for i in range(len(candidates)):path.append(candidates[i])sum = sum + candidates[i]backtracking(candidates[i:],target,sum)sum = sum - candidates[i]path.pop()

    每次递归都要从当前的集合中取数,所以用for循环来完成取数操作,然后每次都要把取得的数值存入path中,sum加上对应的数值,然后递归,注意此时的candidates要变成该元素之后的所有元素组成的集合(包括该元素),然后回溯,将sum减去加上的值,path弹出对应的值。

总体代码为:

def combinationSum(candidates, target):result = []path = []def backtracking(candidates,target,sum):if sum >target:returnif sum == target:path1 = path.copy() # 如果不copy的话,后面修改path的值,result中的path也会改变result.append(path1)returnfor i in range(len(candidates)):path.append(candidates[i])sum = sum + candidates[i]backtracking(candidates[i:],target,sum)sum = sum - candidates[i]path.pop()backtracking(candidates,target,0)return result

2.优化

我们考虑如果我们将candidates从小到大排序后,那么我们从最小的数值按照顺序开始取,这样如果有一次取得的元素和刚好等于或者大于了target,那么后面的该层循环后面的值都不用再取了,因为一定会大于target。
在这里插入图片描述
所以我们可以在逻辑部分加一个判断,如果当sum>target的时候,那么就直接break就好了,不过前提是要先将candidates从小到大排序。

所以修改后的代码为:

def combinationSum(candidates, target):result = []path = []candidates = sorted(candidates)def backtracking(candidates,target,sum):if sum >target:returnif sum == target:path1 = path.copy()result.append(path1)returnfor i in range(len(candidates)):path.append(candidates[i])sum = sum + candidates[i]if sum > target: # 剪枝sum = sum - candidates[i] # 注意判断完后也要回溯path.pop()breakbacktracking(candidates[i:],target,sum)sum = sum - candidates[i]path.pop()backtracking(candidates,target,0)return result

3.总结

剪枝不一定是在原有基础上剪枝,有可能还需要先对集合或者其他要素进行一定操作之后才可以剪枝。


推荐阅读
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 丽江客栈选择问题
    本文介绍了一道经典的算法题,题目涉及在丽江河边的n家特色客栈中选择住宿方案。两位游客希望住在色调相同的两家客栈,并在晚上选择一家最低消费不超过p元的咖啡店小聚。我们将详细探讨如何计算满足条件的住宿方案总数。 ... [详细]
  • 本教程详细介绍了如何使用 TensorFlow 2.0 构建和训练多层感知机(MLP)网络,涵盖回归和分类任务。通过具体示例和代码实现,帮助初学者快速掌握 TensorFlow 的核心概念和操作。 ... [详细]
  • 本文将深入探讨如何在不依赖第三方库的情况下,使用 React 处理表单输入和验证。我们将介绍一种高效且灵活的方法,涵盖表单提交、输入验证及错误处理等关键功能。 ... [详细]
  • 深入剖析 DEX 赛道:从 60 大头部项目看五大趋势
    本文通过分析 60 大头部去中心化交易平台(DEX),揭示了当前 DEX 赛道的五大发展趋势,包括市场集中度、跨链协议、AMM+NFT 结合、新公链崛起以及稳定币和衍生品交易的增长潜力。 ... [详细]
  • 数据结构入门:栈的基本概念与操作
    本文详细介绍了栈这一重要的数据结构,包括其基本概念、顺序存储结构、栈的基本操作(如入栈、出栈、清空栈和销毁栈),以及如何利用栈实现二进制到十进制的转换。通过具体代码示例,帮助读者更好地理解和应用栈的相关知识。 ... [详细]
  • 本文将详细探讨Linux pinctrl子系统的各个关键数据结构,帮助读者深入了解其内部机制。通过分析这些数据结构及其相互关系,我们将进一步理解pinctrl子系统的工作原理和设计思路。 ... [详细]
  • Python处理Word文档的高效技巧
    本文详细介绍了如何使用Python处理Word文档,涵盖从基础操作到高级功能的各种技巧。我们将探讨如何生成文档、定义样式、提取表格数据以及处理超链接和图片等内容。 ... [详细]
  • 本文详细探讨了JavaScript中的作用域链和闭包机制,解释了它们的工作原理及其在实际编程中的应用。通过具体的代码示例,帮助读者更好地理解和掌握这些概念。 ... [详细]
  • 二维几何变换矩阵解析
    本文详细介绍了二维平面上的三种常见几何变换:平移、缩放和旋转。通过引入齐次坐标系,使得这些变换可以通过统一的矩阵乘法实现,从而简化了计算过程。文中不仅提供了理论推导,还附有Python代码示例,帮助读者更好地理解这些概念。 ... [详细]
  • 社交网络中的级联行为 ... [详细]
  • 2018-2019学年第六周《Java数据结构与算法》学习总结
    本文总结了2018-2019学年第六周在《Java数据结构与算法》课程中的学习内容,重点介绍了非线性数据结构——树的相关知识及其应用。 ... [详细]
  • 本文介绍如何使用MFC和ADO技术调用SQL Server中的存储过程,以查询指定小区在特定时间段内的通话统计数据。通过用户界面选择小区ID、开始时间和结束时间,系统将计算并展示小时级的通话量、拥塞率及半速率通话比例。 ... [详细]
  • 本文探讨了如何通过预处理器开关选择不同的类实现,并解决在特定情况下遇到的链接器错误。 ... [详细]
  • LeetCode 690:计算员工的重要性评分
    在解决LeetCode第690题时,我记录了详细的解题思路和方法。该问题要求根据员工的ID计算其重要性评分,包括直接和间接下属的重要性。本文将深入探讨如何使用哈希表(Map)来高效地实现这一目标。 ... [详细]
author-avatar
多米音乐_34492579
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有