118、【回溯算法】leetcode ——40. 组合总和 II:回溯法+剪枝优化(C++版本)

news2025/1/10 21:40:22

题目描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
原题链接:40. 组合总和 II

解题思路

本题的特点是,一个允许结果中出现相同数字,但每个元素仅能被选取一次。结果与结果之间不允许有重复,需要去重。
与 77. 组合(回溯法+剪枝优化) 的相同之处在于选取的元素都是只能被选一次,不同之处在于77里面没有重组元素。
与 39. 组合总和(剪枝优化) 的相同之处在于结果中可能会含有重复元素,但结果与结果之间不能重复。不同之处在于,元素可能被选多次。

  1. 元素只能被选一次:
    考虑了上述个相关题目的思路,依旧设置startIndex = i + 1控制起始位置,保证每个元素仅被选中一次。
  2. 结果集中去重:
    (1)若数组中元素都不相同,则根据startIndex = i + 1可知,每次选取的都为不相同元素,则不会出现重复结果集。
    (2)若数组中含有相同元素,则可能会出现重复的情况是之前已经某一相同元素与其余相同元素组合符合目标之和,当再选取此时的相同元素元素时,会与哪几个其余元素拼凑再一次导致出现重复结果。例如nums = [1, 1, 2], target = 3,则会出现两次选择[1, 2]
    因此,采取的措施是,当第一次遍历到含有相同元素的数时,正常遍历,当第二次遍历到上一次有重复元素的数,跳过该种情况即可。判定条件为i > 0 && candidates[i] == candidates[i - 1],但因为第一次遍历时,递归到下一个数时,也是会有1,但此时第一次遍历时还没结束,不能被跳出。因此需要再设置一个bool型向量visited判定上一个重复元素是否为本轮正被遍历元素,如果正被遍历元素,则继续正常向下遍历,直至完成该次遍历。如果不是,则说明是新一轮遍历,为了去重,直接跳过该次遍历。
class Solution {
public:
    vector<vector<int>> res;
    vector<int> index;    
    void backtracking(vector<int>& candidates, int target, int startIndex, vector<int> path, vector<bool> visited) {
        if(target == 0) {
            res.push_back(path);            
            return ;
        }
        for(int i = startIndex; i < candidates.size(); i++) {
            // 剪枝优化
            if(target - candidates[i] < 0)          return ;
            // i > 0 保证i-1不越界,candidates[i - 1] == candidates[i]判定是否前方有重复元素,visited[i - 1] == false判定为该元素的第一次正被遍历,还是已被遍历过,出现的新一轮遍历
            if(i > 0 && candidates[i - 1] == candidates[i] && visited[i - 1] == false) {             
                // 当该元素已被遍历过,此次为新一轮遍历时,则跳过该种情况
                continue;                
            }            
            visited[i] = true;
            path.push_back(candidates[i]);
            backtracking(candidates, target - candidates[i], i + 1, path, visited);            
            path.pop_back();            
            visited[i] = false;
        }
    }   
    vector<vector<int>> combinationSum2(vector<int>& candidates, int target) {
        vector<int> path;
        vector<bool> visited(candidates.size(), false);
        // 先排序,再剪枝优化
        sort(candidates.begin(), candidates.end());
        backtracking(candidates, target, 0, path, visited);
        return res;
    }
};

2、startIndex判定是否去重

startIndex表示某次遍历的首个元素,根据istartIndex的关系大小,先去重会出现重复元素的结果,然后再让唯一情况加入结果集。

i == startIndex时,说明此时刚开始遍历某一数字。当i > startIndex时,说明已经遍历过前方的元素,正向后续遍历,若此时也出现candidates[i - 1] == candidates[i],则说明会出现重复情况,提前跳出该情况。

class Solution {
public:
    vector<vector<int>> res;
    vector<int> index;    
    void backtracking(vector<int>& candidates, int target, int startIndex, vector<int> path) {
        if(target == 0) {
            res.push_back(path);            
            return ;
        }
        for(int i = startIndex; i < candidates.size(); i++) {
            if(target - candidates[i] < 0)          return ;
            if(i > startIndex && candidates[i - 1] == candidates[i]) {                
                continue;                
            }            
            path.push_back(candidates[i]);
            backtracking(candidates, target - candidates[i], i + 1, path);            
            path.pop_back();            
        }
    }   
    vector<vector<int>> combinationSum2(vector<int>& candidates, int target) {
        vector<int> path;
        sort(candidates.begin(), candidates.end());
        backtracking(candidates, target, 0, path);
        return res;
    }
};

原题链接:40.组合总和II

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/179609.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【JavaSE专栏9】Java 注释知多少

作者主页&#xff1a;Designer 小郑 作者简介&#xff1a;Java全栈软件工程师一枚&#xff0c;来自浙江宁波&#xff0c;负责开发管理公司OA项目&#xff0c;专注软件前后端开发&#xff08;Vue、SpringBoot和微信小程序&#xff09;、系统定制、远程技术指导。CSDN学院、蓝桥云…

斐波那契数列的--------5种算法(又称“兔子数列”)

斐波那契数列&#xff08;Fibonacci sequence&#xff09;&#xff0c;又称黄金分割数列&#xff0c;因数学家莱昂纳多斐波那契&#xff08;Leonardo Fibonacci&#xff09;以兔子繁殖为例子而引入&#xff0c;故又称为“兔子数列”&#xff0c;指的是这样一个数列&#xff1a;…

决策树-剪枝处理

前言&#xff1a;理解《机器学习》P79-83中的决策树剪枝示例。 决策树生成 原始数据集如下所示&#xff0c;前10行为训练集&#xff0c;后7行为验证集&#xff0c;由此数据集可生成如下所示的决策树。 下面解释未进行剪枝操作的决策树为何如上图所示。 不对解释每个结点和分支…

WPF-3D图形

WPF-3D图形 WPF的3D功能可以在不编写任何c#代码的情况下进行绘制&#xff0c;只需要使用xaml即可完成3D图形的渲染。本文主要讲述了WPF-3D中的关键概念&#xff0c; 以及常用到的命中测试、2d控件如何在3D对象中进行渲染&#xff0c;除此之外&#xff0c;还演示了如何导入外部…

InstanceNorm LayerNorm

InstanceNorm && LayerNorm author: SUFEHeisenberg date: 2023/01/26 先说结论: 将Transformer类比于RNN&#xff1a;一个token就是一层layer&#xff0c;对一整句不如token有意义原生Bert代码或huggingface中用的都是InstanceNorm instead of LayerNorm&#xff…

【AAAI2023】Head-Free Lightweight Semantic Segmentation with Linear Transformer

论文&#xff1a;【AAAI2023】Head-Free Lightweight Semantic Segmentation with Linear Transformer 代码&#xff1a;https://github.com/dongbo811/AFFormer 这是来自阿里巴巴的工作&#xff0c;作者构建了一个轻量级的Transformer网络用于语义分割&#xff0c;主要有两点…

发现下属的学历造假,但是他的工作能力又很强,该开除他吗?

在职场上混&#xff0c;学历是敲门砖还是定音锤呢&#xff1f;一位网友问&#xff1a;发现下属的学历造假&#xff0c;但是他的工作能力又很强&#xff0c;该开除他吗?有人觉得一定要开除&#xff0c;这就是钻空子&#xff0c;受影响最大的人不是他&#xff0c;而是那些真才实…

上采样与下采样

数据分析中的上采样和下采样 背景&#xff1a; 在分类问题中&#xff0c;由于各种原因&#xff0c;我们所获取到的数据集很容易出现正负样本的不平衡&#xff0c;或者某些数据特别多&#xff0c;有些数据则特别少&#xff0c;在这样的数据集中&#xff0c;进行训练&#xff0c…

OpenCV直方图Java 演示程序

直方图Java 演示程序以下文件编码为utf-8 为佳。代码文件名&#xff1a;OpenCvMain.javapackage org.opencv;import java.net.URL;import java.util.LinkedList;import java.util.List;import org.opencv.core.Core;import org.opencv.core.CvType;import org.opencv.core.Mat;…

Linux常用命令——setpci命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) setpci 查询和配置PCI设备的使用工具 补充说明 setpci命令是一个查询和配置PCI设备的使用工具。 语法 setpci(选项)(参数)选项 -v&#xff1a;显示指令执行的细节信息&#xff1b; -f&#xff1a;当没有任何…

Opencv形态学操作——腐蚀、膨胀、梯度、开运算、闭运算、礼帽、黑帽(附案例详细讲解及可执行代码)

Opencv形态学操作 腐蚀膨胀梯度开运算闭运算礼帽黑帽总结腐蚀 在地理或者化学中,我们学习过腐蚀,是指在某种 作用下产生损耗与破坏的过程。你也可以理解为减肥。在Opencv中,腐蚀操作可以使白色轮廓变小,也就是说可以去除一些白色的噪声。 如果你接触过卷积核的话,腐蚀就更…

【JavaSE专栏8】运算符、表达式和语句

作者主页&#xff1a;Designer 小郑 作者简介&#xff1a;Java全栈软件工程师一枚&#xff0c;来自浙江宁波&#xff0c;负责开发管理公司OA项目&#xff0c;专注软件前后端开发&#xff08;Vue、SpringBoot和微信小程序&#xff09;、系统定制、远程技术指导。CSDN学院、蓝桥云…

盖子的c++小课堂——第十三讲:二维数组

前言 过了几天了&#xff0c;终于有时间更新了&#xff0c;有个通知&#xff0c;以后我不用颜色区分了&#xff0c;不然换了背景看不见&#xff0c;理解一下&#xff0c;蟹蟹~~ 举例 作者&#xff1a;一下是某次奥运会的奖牌榜&#xff0c;你知道如何储存奖牌榜吗~~ 粉丝&am…

机器学习中软投票和硬投票的不同含义和理解

设置一个场景&#xff0c;比如对于今天音乐会韩红会出现的概率三个人三个观点 A&#xff1a;韩红出现的概率为47% B&#xff1a;韩红出现的概率为57% C&#xff1a;韩红出现的概率为97% 软投票&#xff1a;软投票会认为韩红出现的概率为1/3*(47%57%97%)67% 硬投票&#xff1a;…

“子序列问题”系列总结,一文读懂(Java实现)

目录 前言 一、最长递增子序列 1.1、dp定义 1.2、递推公式 1.3、初始化 1.4、注意 1.5、解题代码 二、最长连续递增序列 2.1、分析 2.2、解题代码 三、最长重复子数组 3.1、dp定义 3.2、递推公式 3.3、初始化 3.4、解题代码 四、最长公共子序列 4.1、分析 4.2…

Opencv项目实战:20 单手识别数字0到5

目录 0、项目介绍 1、效果展示 2、项目搭建 3、项目代码展示 HandTrackingModule.py Figures_counter.py 4、项目资源 5、项目总结 0、项目介绍 今天要做的是单手识别数字0到5&#xff0c;通过在窗口展示&#xff0c;实时的展示相应的图片以及文字。 在网上找了很久的…

硬核来袭!!!一篇文章教你入门Python爬虫网页解析神器——BeautifulSoup详细讲解

文章目录一、BeautifulSoup介绍二、安装三、bs4数据解析的原理四、bs4 常用的方法和属性1、BeautifulSoup构建1.1 通过字符串构建1.2 从文件加载2、BeautifulSoup四种对象2.1 Tag对象2.2 NavigableString对象2.3 BeautifulSoup对象2.4 Comment对象五、contents、children与desc…

springboot自定义拦截器的简单使用和一个小例子

springboot自定义拦截器的使用1. 自定义拦截器2. 拦截器登录验证的小demo2.1 配置pom.xml2.2 创建User的bean组件2.3 创建需要的表单页面以及登录成功的页面2.4 编写controller映射关系2.5 自定义拦截器类&#xff0c;实现intercepetor接口2.6注册添加拦截器&#xff0c;自定义…

【SpringCloud】Nacos集群搭建

集群结构图官方给出的Nacos集群图如下&#xff1a;其中包含3个nacos节点&#xff0c;然后一个负载均衡器代理3个Nacos。这里负载均衡器可以使用nginx。我们接下来要尝试 Nacos集群搭建&#xff0c;效果图如下所示&#xff1a;三个nacos节点的地址&#xff1a;节点ipportnacos1l…

二、Java框架之Spring注解开发

文章目录1. IOC/DI注解开发1.1 Component注解ComponentController Service Repository1.2 纯注解开发模式1.3 注解开发bean管理ScopePostConstruct PreDestroy1.4 注解开发依赖注入Autowired QualifierValuePropertySource1.5 第三方bean管理Beanimport&#xff08;多个Config类…