[Algorithm][回溯][组合][目标和][组合总和]详细讲解

news2024/10/4 22:30:04

目录

  • 1.组合
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现
  • 2.目标和
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现
  • 3.组合总和
    • 1.题目链接
    • 2.算法原理详解
    • 3.代码实现


1.组合

1.题目链接

  • 组合

2.算法原理详解

  • 思路:每次都只选一个数,此后只能选它后面的数
  • 函数设计
    • 全局变量
      • vector<vector<int>> ret
      • vector<int> path
    • DFS()设计:void DFS(nums, pos)
    • 递归出口path.size() == k
    • 剪枝:控制参数,每次从此位置下一个位置开始递归
      请添加图片描述

3.代码实现

class Solution 
{
    int _n;
    int _k;

    vector<vector<int>> ret;
    vector<int> path;
public:
    vector<vector<int>> combine(int n, int k) 
    {
        _n = n;
        _k = k;
        DFS(1);
        return ret;
    }

    void DFS(int start)
    {
        if(path.size() == _k)
        {
            ret.push_back(path);
        }

        // 递归 + 剪枝
        for(int i = start; i <= _n; i++)
        {
            path.push_back(i);
            DFS(i + 1);
            path.pop_back(); // 回溯,恢复现场
        }
    }
};

2.目标和

1.题目链接

  • 目标和

2.算法原理详解

  • 本题与子集逻辑几乎相同
  • 本题会实现两种代码,可以通过这两种代码来感受:回溯的两种做法
    • path是全局变量的时候
      • 本题可能会超时
    • path作为参数
      • 此时编译器/代码会代为回溯,每次回溯都省去了一次加/减运算,故效率有所提高
        请添加图片描述

3.代码实现

// v1.0 效率低,可能会超时
class Solution 
{
    int ret = 0;
    int path = 0;
    int _target = 0;
public:
    int findTargetSumWays(vector<int>& nums, int target) 
    {
        _target = target;
        DFS(nums, 0);
        return ret;
    }

    void DFS(vector<int>& nums, int pos)
    {
        if(pos == nums.size())
        {
            if(path == _target)
            {
                ret++;
            }

            return;
        }

        // 加
        path += nums[pos];
        DFS(nums, pos + 1);
        path -= nums[pos]; // 回溯,恢复现场

        // 减
        path -= nums[pos];
        DFS(nums, pos + 1);
        path += nums[pos]; // 回溯,恢复现场
    }
};
--------------------------------------------------------------------------
// v2.0,效率有所改善
class Solution 
{
    int ret = 0;
    int _target = 0;
public:
    int findTargetSumWays(vector<int>& nums, int target) 
    {
        _target = target;
        DFS(nums, 0, 0);
        return ret;
    }

    void DFS(vector<int>& nums, int pos, int path)
    {
        if(pos == nums.size())
        {
            if(path == _target)
            {
                ret++;
            }

            return;
        }

        // 加
        DFS(nums, pos + 1, path + nums[pos]);

        // 减
        DFS(nums, pos + 1, path - nums[pos]);
    }
};

3.组合总和

1.题目链接

  • 组合总和

2.算法原理详解

  • 思路一:每次都只选一个数,此后只能选它及它后面的数

    • 函数设计
      • 全局变量
        • vector<vector<int>> ret
        • vector<int> path
      • DFS()设计:void DFS(nums, pos, sum)
      • 递归出口sum == _target || (sum > _target || pos == nums.size())
      • 回溯:通过sum控制回溯
      • 剪枝:控制pos参数,每次从此位置开始递归
        请添加图片描述
  • 思路二:每次枚举一个数,出现几次

    • 函数设计
      • 全局变量
        • vector<vector<int>> ret
        • vector<int> path
      • DFS()设计:void DFS(nums, pos, sum)
      • 递归出口sum == _target || (sum > _target || pos == nums.size())
      • 回溯:通过sum控制回溯
      • 剪枝:控制pos参数,每次从此位置开始递归
        请添加图片描述

3.代码实现

// v1.0 每次都只选一个数,此后只能选它及它后面的数
class Solution 
{
    int _target;
    vector<int> path;
    vector<vector<int>> ret;
public:
    vector<vector<int>> combinationSum(vector<int>& nums, int target) 
    {
        _target = target;
        DFS(nums, 0, 0);
        return ret;
    }
    
    void DFS(vector<int>& nums, int pos, int sum)
    {
        if(sum == _target)
        {
            ret.push_back(path);
            return;
        }
        
        if(sum > _target || pos == nums.size())
        {
            return;
        }
        
        // 递归决策 + 剪枝
        for(int i = pos; i < nums.size(); i++)
        {
            path.push_back(nums[i]);
            DFS(nums, i, sum + nums[i]);
            path.pop_back(); // 回溯,恢复现场
        }
    }
};
--------------------------------------------------------------------------
// v2.0 每次枚举一个数,出现几次
class Solution 
{
    int _target;
    vector<int> path;
    vector<vector<int>> ret;
public:
    vector<vector<int>> combinationSum(vector<int>& nums, int target) 
    {
        _target = target;
        DFS(nums, 0, 0);
        return ret;
    }
    
    void DFS(vector<int>& nums, int pos, int sum)
    {
        if(sum == _target)
        {
            ret.push_back(path);
            return;
        }
        
        if(sum > _target || pos == nums.size())
        {
            return;
        }
        
        // 枚举个数 + 剪枝
        for(int i = 0; i * nums[pos] + sum <= _target; i++)
        {
            if(i)
            {
                path.push_back(nums[pos]);    
            }
            
            DFS(nums, pos + 1, i * nums[pos] + sum);
        }
        
        // 回溯,恢复现场
        for(int i = 1; i * nums[pos] + sum <= _target; i++)
        {
            path.pop_back();
        }
    }
};

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

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

相关文章

全面提升数据采集效率:亮数据产品的应用与评估详解

全面提升数据采集效率&#xff1a;亮数据产品的应用与评估详解 文章目录 全面提升数据采集效率&#xff1a;亮数据产品的应用与评估详解背景应用场景&#xff1a;平台首页信息抓取准备评测素材详细的产品使用和评测流程产品介绍亮数据的IP代理服务亮数据的爬虫工具及采集技术 注…

elasticsearch使用Ngram实现任意位数手机号搜索

文章目录 Ngram自定义分词案例实战问题拆解 Ngram分词器定义Ngram分词定义Ngram分词示例Ngram分词应用场景 Ngram分词实战 Ngram自定义分词案例 当对keyword类型的字段进行高亮查询时&#xff0c;若值为123asd456&#xff0c;查询sd4&#xff0c;则高亮结果是&#xff1c;em&a…

项目管理-案例重点知识(整合管理)

项目管理&#xff1a;每天进步一点点~ 活到老&#xff0c;学到老 ヾ(◍∇◍)&#xff89;&#xff9e; 何时学习都不晚&#xff0c;加油 一、整合管理 案例重点 重点内容&#xff1a; &#xff08;1&#xff09;项目章程内容和作用 &#xff08;2&#xff09;项目管理计划…

乡村振兴的农业科技创新:加大农业科技投入,推广农业科技成果,提升农业科技创新水平,推动美丽乡村农业现代化

一、引言 随着全球化和信息化时代的到来&#xff0c;农业作为国民经济的基础&#xff0c;其现代化进程日益受到关注。在乡村振兴战略的大背景下&#xff0c;农业科技创新成为推动乡村经济转型升级、实现农业现代化的关键力量。本文旨在探讨如何通过加大农业科技投入、推广农业…

【PB案例学习笔记】-02 目录浏览器

写在前面 这是PB案例学习笔记系列文章的第二篇&#xff0c;该系列文章适合具有一定PB基础的读者&#xff0c; 通过一个个由浅入深的编程实战案例学习&#xff0c;提高编程技巧&#xff0c;以保证小伙伴们能应付公司的各种开发需求。 文章中设计到的源码&#xff0c;小凡都上…

蓝桥杯单片机之模块代码《串口发数据》

过往历程 历程1&#xff1a;秒表 历程2&#xff1a;按键显示时钟 历程3&#xff1a;列矩阵按键显示时钟 历程4&#xff1a;行矩阵按键显示时钟 历程5&#xff1a;新DS1302 历程6&#xff1a;小数点精确后两位ds18b20 历程7&#xff1a;35定时器测量频率 历程8&#xff…

初识鸿蒙之ArkTS基础

前言 学习一种应用程序开发&#xff0c;需要从这种程序的开发语言开始&#xff0c;比如说Android开发从入门到放弃&#xff0c;肯定是从Java基础或者是Kotlin语言基础开始学习的&#xff0c;IOS程序开发也肯定是从object-c开始学习的。鸿蒙软件开发也不例外&#xff0c;如果做…

平衡三进制小数详解与进制转换

标准三进制是“逢三进一&#xff0c;退一还三”的机制&#xff0c;平衡三进制与之类似&#xff0c;但就是偏移了一下变得对称了&#xff0c;平衡三进制与标准三进制可以相互转换&#xff0c;但这样显得有点多余了&#xff0c;所以这里只讲平衡三进制与十进制的转换。 数字系统的…

meshlab: pymeshlab合并多个物体模型并保存(flatten visible layers)

一、关于环境 请参考&#xff1a;pymeshlab遍历文件夹中模型、缩放并导出指定格式-CSDN博客 二、关于代码 本文所给出代码仅为参考&#xff0c;禁止转载和引用&#xff0c;仅供个人学习。 本文所给出的例子是https://download.csdn.net/download/weixin_42605076/89233917中的…

【微记录】dmidecode是干什么的?常用来做什么?如何查看系统支持的PCIe版本号(本质:标准,Desktop Management Interface)

是什么 dmidecode 是一个在 Linux 系统提取硬件信息的命令行工具。DMI 代表桌面管理接口&#xff08;Desktop Management Interface&#xff09;&#xff0c;是一种标准&#xff0c;收集桌面计算机的硬件信息&#xff0c;包括系统制造商、序列号、BIOS 信息、系统资产标签等。…

风电功率预测 | 基于PSO-BP神经网络实现风电功率预测(附matlab完整源码)

风电功率预测 风电功率预测完整代码风电功率预测 基于粒子群优化算法(Particle Swarm Optimization, PSO)的BP神经网络是一种常见的方法,用于实现风电功率预测。下面是一个基于PSO-BP神经网络实现风电功率预测的一般步骤: 数据准备:收集与风电场发电功率相关的数据,包括…

C语言----斐波那契数列(附源代码)

各位看官们好&#xff0c;当我写了上一篇博客杨辉三角后&#xff0c;有一些看官叫我讲一下斐波那契数列。对于这个大家应该是有了解的。最简单的规律就是f(n)f(n-2)f(n-1)。就是当前是前两项之和&#xff0c;然后下标1和0都是1.从第三项开始计算的。那么我们知道规律&#xff0…

旧衣服回收小程序:探索旧衣回收市场的创新发展

每年我国就有将近800万吨旧衣服&#xff0c;在生活水平的日益提高下&#xff0c;这个数字也在逐渐增加。目前&#xff0c;我国旧衣回收的产业链也在完善中&#xff0c;旧衣服出口贸易逐年增加&#xff0c;市场发展空间不断扩大。此外&#xff0c;旧衣回收市场投入低、风险小、利…

SystemC学习使用记录

一、概述 对于复杂的片上系统&#xff0c;在进行RTL编码前&#xff0c;需进行深入的系统级仿真&#xff0c;以确认设计的体系结构是否恰当、总线是否能满足吞吐量和实现性要求以及存储器是否浪费&#xff0c;所进行的这些仿真要求在芯片的仿真模型上运行大量的软件&#xff0c…

PLL-分频器

概念 分频器的性能一般用四个参数来规定:(1)分频比&#xff0c;(2)最大允许输入频率fmax&#xff0c;(3)功耗&#xff0c;(4)最小允许输入电压摆幅(也叫“灵敏度”)。虽然分频器的相位噪声也很重要&#xff0c;但在大多数情况下它可以忽略不计。 把一般分频器的输入灵敏度画成…

微信支付商户的“商家转账到零钱”产品快速开通指南

微信支付商户的“商家转账到零钱”功能为商家提供了便捷的转账途径&#xff0c;尤其适用于费用报销、员工福利发放、合作伙伴货款或分销返佣等多种场景。那么&#xff0c;如何快速开通这一功能呢&#xff0c;需要快速开通的商户可以联系小编。 首先&#xff0c;确保你的企业已经…

C++基础与深度解析 | 表达式 | 操作符

文章目录 一、表达式基础1.表达式的值类别2.表达式的类型转换 二、表达式详述1.算术操作符2.逻辑与关系操作符3.位操作符4.赋值操作符5.自增与自减运算符6.其他操作符三、C17对表达式的求值顺序的限定 一、表达式基础 表达式由一到多个操作数组成&#xff0c;可以求值并 ( 通常…

Node.js 学习笔记 express框架

express express 使用express下载express 初体验 express 路由什么是路由1路由的使用验证的方法 2获取请求报文参数3获取路由参数4响应设置响应报文 express 中间件5中间件全局中间件路由中间件 6静态资源中间件注意事项案例 7请求体数据8防盗链实现防盗链 9路由模块化router E…

Linux: Make工具以及Makefile文件

make工具 人们通常利用 make 工具来自动完成编译工作。这些工作包括&#xff1a;如果仅修改了某几个源文件&#xff0c;则只重新编译这几个源文件&#xff1b;如果某个头文件被修改了&#xff0c;则重新编译所有包含该头文件的源文件。利用这种自动编译可大大简化开发工作&…

TreeMap详解:Java 有序 Map 原理与实现

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。运营社区&#xff1a;C站/掘金/腾讯云&#xff1b;欢迎大家常来逛逛 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互相学习&#xff0c;一…