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



推荐阅读
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • Java 类成员初始化顺序与数组创建
    本文探讨了Java中类成员的初始化顺序、静态引入、可变参数以及finalize方法的应用。通过具体的代码示例,详细解释了这些概念及其在实际编程中的使用。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 在金融和会计领域,准确无误地填写票据和结算凭证至关重要。这些文件不仅是支付结算和现金收付的重要依据,还直接关系到交易的安全性和准确性。本文介绍了一种使用C语言实现小写金额转换为大写金额的方法,确保数据的标准化和规范化。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 机器学习中的相似度度量与模型优化
    本文探讨了机器学习中常见的相似度度量方法,包括余弦相似度、欧氏距离和马氏距离,并详细介绍了如何通过选择合适的模型复杂度和正则化来提高模型的泛化能力。此外,文章还涵盖了模型评估的各种方法和指标,以及不同分类器的工作原理和应用场景。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
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社区 版权所有