热门标签 | 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.总结

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


推荐阅读
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 尽管使用TensorFlow和PyTorch等成熟框架可以显著降低实现递归神经网络(RNN)的门槛,但对于初学者来说,理解其底层原理至关重要。本文将引导您使用NumPy从头构建一个用于自然语言处理(NLP)的RNN模型。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 导航栏样式练习:项目实例解析
    本文详细介绍了如何创建一个具有动态效果的导航栏,包括HTML、CSS和JavaScript代码的实现,并附有详细的说明和效果图。 ... [详细]
  • 本文基于刘洪波老师的《英文词根词缀精讲》,深入探讨了多个重要词根词缀的起源及其相关词汇,帮助读者更好地理解和记忆英语单词。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 本文探讨了如何优化和正确配置Kafka Streams应用程序以确保准确的状态存储查询。通过调整配置参数和代码逻辑,可以有效解决数据不一致的问题。 ... [详细]
  • 机器学习中的相似度度量与模型优化
    本文探讨了机器学习中常见的相似度度量方法,包括余弦相似度、欧氏距离和马氏距离,并详细介绍了如何通过选择合适的模型复杂度和正则化来提高模型的泛化能力。此外,文章还涵盖了模型评估的各种方法和指标,以及不同分类器的工作原理和应用场景。 ... [详细]
  • 本文探讨了如何在编程中正确处理包含空数组的 JSON 对象,提供了详细的代码示例和解决方案。 ... [详细]
  • 探索1000以内的完美数:因数和等于自身
    本文探讨了如何在1000以内找到所有完美数,即一个数的因数(不包括自身)之和等于该数本身。例如,6是一个完美数,因为1 + 2 + 3 = 6。通过编程实现这一过程,可以更好地理解完美数的特性。 ... [详细]
  • 本题通过将每个矩形视为一个节点,根据其相对位置构建拓扑图,并利用深度优先搜索(DFS)或状态压缩动态规划(DP)求解最小涂色次数。本文详细解析了该问题的建模思路与算法实现。 ... [详细]
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社区 版权所有