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

LeetCode40:组合总和II——解析与优化算法

本文详细解析了LeetCode第40题“组合总和II”的算法思路与优化方法。题目要求从给定的数组`candidates`中找出所有可能的组合,使得这些组合中的数字之和等于目标值`target`。文章不仅介绍了基本的回溯算法,还探讨了如何通过剪枝技术提高算法效率,以应对大规模数据输入。

这道题是LeetCode里的第40道题。

题目要求:

给定一个数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。

candidates 中的每个数字在每个组合中只能使用一次。

说明:

所有数字(包括目标数)都是正整数。

解集不能包含重复的组合。

示例 1:

输入: candidates = [10,1,2,7,6,1,5], target = 8,

所求解集为:

[

[1, 7],

[1, 2, 5],

[2, 6],

[1, 1, 6]

]

示例 2:

输入: candidates = [2,5,2,1,2], target = 5,

所求解集为:

[

[1,2,2],

[5]

]

解法和第39题几乎一样,区别在于本题每一个数字只能使用一次,但数字在数组中是可以重复的。同样,还是使用回溯剪枝法,先对 candidates数组元素排序。排序后进行循环递归。具体代码如下:

提交代码:

class Solution {

public:

vector> res;

vector ans;

void getres(vector& candidates,int target,int k,vector ans){

int size=candidates.size();

for(int i=k;i

if(target-candidates[i]>0){

if(i>k&&candidates[i]==candidates[i-1])continue;

ans.push_back(candidates[i]);

getres(candidates,target-candidates[i],i+1,ans);

ans.pop_back();

}

else if(target-candidates[i]<0){return;}

else{

ans.push_back(candidates[i]);

res.push_back(ans);

ans.pop_back();

return;

}

}

return;

}

vector> combinationSum2(vector& candidates, int target) {

sort(candidates.begin(),candidates.end());

getres(candidates,target,0,ans);

return res;

}

};

代码和第39题几乎一样&#xff0c;只改了两个地方&#xff0c;第一个是第11行&#xff0c;参数由 i 改为 i&#43;1&#xff0c;这是能想到的确保每一个数字只使用一次的改法。但是这样改还并不完全&#xff0c;因为最终答案会有重复的解答。第二个改法纠结我好久&#xff1a;最笨的想法是先把解保存在一个 set 集合中&#xff0c;然后在转入 vector> 中&#xff0c;但这样太慢了&#xff0c;不想用。那么是什么原因造成解的重复呢&#xff1f;原因就是 candidates 数组中包含着重复数&#xff0c;例如&#xff1a;

示例1&#xff1a;candidates &#61; [10,1,2,7,6,1,5]&#xff0c;target &#61; 8&#xff0c;标准解答&#xff1a;[[1,2,5],[1,7],[1,1,6],[2,6]]

没有第九行代码前的解答&#xff1a;[[1,1,6],[1,2,5],[1,7],[1,2,5],[1,7],[2,6]]&#xff0c;其中第二个和第四个重复&#xff0c;第三个和第五个重复。因为数组中有两个1&#xff0c;这两个1分别组合了一次&#xff0c;造成了重复。知道原因了&#xff0c;就好解决&#xff0c;首先想到的是加入条件candidates[i]&#61;&#61;candidates[i-1]当前后元素相等时&#xff0c;直接跳过本次循环&#xff0c;但问题又来了&#xff1a;[1,1,6]这个解&#xff0c;前后元素相等&#xff0c;但不重复&#xff0c;怎么办&#xff1f;这个我想了好久&#xff0c;最后看评论&#xff1a;因为当前层&#xff0c;如果再取下一个一样的数的话&#xff0c;就会造成重复&#xff0c;但是在下一层加入就不会&#xff0c;因为当前层是替换&#xff0c;如果拿一个一样的替换肯定会重复。再加入条件&#xff1a;i>k 就能保证既不会缺少[1,1,6]这个解&#xff0c;也保证了解的互异。所以第九行加入代码&#xff1a;if(i>k&&candidates[i]&#61;&#61;candidates[i-1])continue;&#xff0c;然后剩下的剪枝第39题都有。

提交结果&#xff1a;

个人总结&#xff1a;

本题和第39题不同&#xff0c;难度也体现在最后答案的重复上。回溯法的本质还是递归&#xff0c;递归最难的地方就是容易弄混循环和递归层数&#xff0c;需要画图仔细分析&#xff0c;递归循环是个树图&#xff0c;画图也好容易分析。最后这题也可以不用排序&#xff0c;因为数字只使用一次&#xff0c;但是结果是会有重复的&#xff0c;而且重复的是无规则的&#xff0c;效率低。



推荐阅读
  • 贡献转移在计算每个元素的作用的时候,我们可以通过反向枚举作用效果,添加到作用元素的身上,这种方法叫做贡献转移。更正式的说, ... [详细]
  • 题面:P3178[HAOI2015]树上操作好像其他人都嫌这道题太容易了懒得讲,好吧那我讲。题解:第一个操作和第二个操作本质上是一样的&# ... [详细]
  • 本文详细介绍了在PHP中如何获取和处理HTTP头部信息,包括通过cURL获取请求头信息、使用header函数发送响应头以及获取客户端HTTP头部的方法。同时,还探讨了PHP中$_SERVER变量的使用,以获取客户端和服务器的相关信息。 ... [详细]
  • This article explores the process of integrating Promises into Ext Ajax calls for a more functional programming approach, along with detailed steps on testing these asynchronous operations. ... [详细]
  • java datarow_DataSet  DataTable DataRow 深入浅出
    本篇文章适合有一定的基础的人去查看,最好学习过一定net编程基础在来查看此文章。1.概念DataSet是ADO.NET的中心概念。可以把DataSet当成内存中的数据 ... [详细]
  • Web开发实践:创建连连看小游戏
    本文详细介绍了如何在Web环境中开发一款连连看小游戏,适合初学者和技术爱好者参考。通过本文,您将了解游戏的基本结构、连线算法以及实现方法。 ... [详细]
  • 在AngularJS中,有时需要在表单内包含某些控件,但又不希望这些控件导致表单变为脏状态。例如,当用户对表单进行修改后,表单的$dirty属性将变为true,触发保存对话框。然而,对于一些导航或辅助功能控件,我们可能并不希望它们触发这种行为。 ... [详细]
  • 本文详细介绍了Socket在Linux内核中的实现机制,包括基本的Socket结构、协议操作集以及不同协议下的具体实现。通过这些内容,读者可以更好地理解Socket的工作原理。 ... [详细]
  • 探索CNN的可视化技术
    神经网络的可视化在理论学习与实践应用中扮演着至关重要的角色。本文深入探讨了三种有效的CNN(卷积神经网络)可视化方法,旨在帮助读者更好地理解和优化模型。 ... [详细]
  • HDU 2537 键盘输入处理
    题目描述了一个名叫Pirates的男孩想要开发一款键盘输入软件,遇到了大小写字母判断的问题。本文提供了该问题的解决方案及实现方法。 ... [详细]
  • 本文介绍了使用Python和C语言编写程序来计算一个给定数值的平方根的方法。通过迭代算法,我们能够精确地得到所需的结果。 ... [详细]
  • 本文提供了一个关于AC自动机(Aho-Corasick Algorithm)的详细解析与实现方法,特别针对P3796题目进行了深入探讨。文章不仅涵盖了AC自动机的基本概念,还重点讲解了如何通过构建失败指针(fail pointer)来提高字符串匹配效率。 ... [详细]
  • LeetCode 102 - 二叉树层次遍历详解
    本文详细解析了LeetCode第102题——二叉树的层次遍历问题,提供了C++语言的实现代码,并对算法的核心思想和具体步骤进行了深入讲解。 ... [详细]
  • 本文详细介绍了在MyBatis框架中如何通过#和$两种方式来传递SQL查询参数。使用#方式可以提高执行效率,而使用$则有助于在复杂SQL语句中更好地查看日志。此外,文章还探讨了不同场景下的参数传递方法,包括实体对象、基本数据类型以及混合参数的使用。 ... [详细]
  • 本文探讨了在已知最终数组尺寸不会超过5000x10的情况下,如何利用预分配和调整大小的方法来优化Numpy数组的创建过程,以提高性能并减少内存消耗。 ... [详细]
author-avatar
琼慧彦霖顺南
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有