热门标签 | 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;效率低。



推荐阅读
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • Splay Tree 区间操作优化
    本文详细介绍了使用Splay Tree进行区间操作的实现方法,包括插入、删除、修改、翻转和求和等操作。通过这些操作,可以高效地处理动态序列问题,并且代码实现具有一定的挑战性,有助于编程能力的提升。 ... [详细]
  • 本文探讨了如何优化和正确配置Kafka Streams应用程序以确保准确的状态存储查询。通过调整配置参数和代码逻辑,可以有效解决数据不一致的问题。 ... [详细]
  • 机器学习中的相似度度量与模型优化
    本文探讨了机器学习中常见的相似度度量方法,包括余弦相似度、欧氏距离和马氏距离,并详细介绍了如何通过选择合适的模型复杂度和正则化来提高模型的泛化能力。此外,文章还涵盖了模型评估的各种方法和指标,以及不同分类器的工作原理和应用场景。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文探讨了如何在编程中正确处理包含空数组的 JSON 对象,提供了详细的代码示例和解决方案。 ... [详细]
  • 本实验主要探讨了二叉排序树(BST)的基本操作,包括创建、查找和删除节点。通过具体实例和代码实现,详细介绍了如何使用递归和非递归方法进行关键字查找,并展示了删除特定节点后的树结构变化。 ... [详细]
  • 本教程涵盖OpenGL基础操作及直线光栅化技术,包括点的绘制、简单图形绘制、直线绘制以及DDA和中点画线算法。通过逐步实践,帮助读者掌握OpenGL的基本使用方法。 ... [详细]
  • 尽管使用TensorFlow和PyTorch等成熟框架可以显著降低实现递归神经网络(RNN)的门槛,但对于初学者来说,理解其底层原理至关重要。本文将引导您使用NumPy从头构建一个用于自然语言处理(NLP)的RNN模型。 ... [详细]
  • 深入理解Java泛型:JDK 5的新特性
    本文详细介绍了Java泛型的概念及其在JDK 5中的应用,通过具体代码示例解释了泛型的引入、作用和优势。同时,探讨了泛型类、泛型方法和泛型接口的实现,并深入讲解了通配符的使用。 ... [详细]
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • 使用GDI的一些AIP函数我们可以轻易的绘制出简 ... [详细]
  • 汇编语言等号伪指令解析:探究其陡峭的学习曲线
    汇编语言以其独特的特性和复杂的语法结构,一直被认为是编程领域中学习难度较高的语言之一。本文将探讨汇编语言中的等号伪指令及其对初学者带来的挑战,并结合社区反馈分析其学习曲线。 ... [详细]
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社区 版权所有