蓝桥杯算法竞赛系列第十章·nSum问题的代码框架

news2024/12/26 22:40:14

在这里插入图片描述

你好,我是安然无虞。

文章目录

  • 一、两数之和
    • 变形题
  • 二、三数之和
  • 三、四数之和

在这里插入图片描述

首先,何为nSum问题呢?

nSum问题其实就是给你一个数组nums和一个目标和target,让我们从nums数组中选择n个数,使得这些数字之和等于target。

由此衍生出了两数之和,三数之和,四数之和……

今天可借用一个算法框架,求解100Sum也不不在话下!

一、两数之和

题目链接:两数之和II-输入有序数组

题目要求:

给你一个下标从 1 开始的整数数组 numbers ,该数组已按 非递减顺序排列 ,请你从数组中找出满足相加之和等于目标数 target 的两个数。如果设这两个数分别是 numbers[index1]numbers[index2] ,则 1 <= index1 < index2 <= numbers.length

以长度为 2 的整数数组 [index1, index2] 的形式返回这两个整数的下标 index1index2

你可以假设每个输入 只对应唯一的答案 ,而且你 不可以 重复使用相同的元素。

你所设计的解决方案必须只使用常量级的额外空间。

简单来说就是:

给我们一个数组和一个目标值,让我们找到和为目标值的两个数的下标。

看起来好像很简单,雀实不难。

解题思路:

首先对数组进行排序,因为本题已经是有序的了,所以不同排序了(手动狗头)

然后就是定义双指针,一个在前一个在后,比较两数之和,就行了。

代码详解:

class Solution {
public:
    vector<int> twoSum(vector<int>& numbers, int target) 
    {
        // 定义前后指针
        int left = 0, right = numbers.size() - 1;

        while(left < right)
        {
            // 前后两个数之和
            int sum = numbers[left] + numbers[right];
						// 根据 sum 和 target 的比较,移动左右指针
            if(sum < target)
            {
                left++;
            }
            else if (sum > target)
            {
                right--;
            }
            else if(sum == target)
            {
                return {left + 1, right + 1};
            }
        }

        return {-1, -1};
    }
};

变形题

下面这这道题进行变形,变得更困难一点:

nums 中可能有多对儿元素之和都等于 target,请你的算法返回所有和为 target 的元素对儿,其中不能出现重复

还数签名如下:

vector<vector<int>> twoSumTarget(vector<int>& nums, int target);

对于这个修改后的问题,关键难点在于现在可能有多个和为target 的数对,还不能重复,比如[1, 3]和[3, 1]就是重复的。

首先解题思想肯定还是和上面一样:排序 + 双指针

while (lo < hi) 
{
    int sum = nums[lo] + nums[hi];
    // 记录索引 lo 和 hi 最初对应的值
    int left = nums[lo], right = nums[hi];
    if (sum < target)      
      	lo++;
    else if (sum > target)
      	hi--;
    else if (sum == target) 
    {
      	// 解题的关键就在于sum == target的情况,这里前后指针需要跳过所有重复的元素
        res.push_back({left, right});
        // 跳过所有重复的元素
        while (lo < hi && nums[lo] == left) lo++;
        while (lo < hi && nums[hi] == right) hi--;
    }
}

OK,到这里,一个通用的twoSum函数就算是写出来了,注意理解哦,因为整个nSum问题都会用到这个函数。

二、三数之和

题目链接:三数之和

在这里插入图片描述

解题思路:

题目要求我们在数组nums中找到和为0的三个数,也就是说这里的n是3,target是0

其实说到底还是穷举,现在要求我们找到和为target的三个数,对于第一个数来说,nums数组中的所有数字都可能是,但是只要第一个数字确定了,剩下的两个数可以是什么呢 ?其实也就是和为target - nums[i]的两个数,这样一来,又回到了twoSum问题本身。

OK,请看下面代码:

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) 
    {
        // 先对数组进行排序
        sort(nums.begin(), nums.end());
        
        // 为了更好的复用,直接封装成为nSum函数
        return nSum(nums, 3, 0, 0);
    }

    vector<vector<int>> nSum(vector<int>& nums, int n, int start, int target)
    {
        int sz = nums.size();

        vector<vector<int>> res;

        // 注意边界
        if(n < 2 || n > sz)
        {
            return res;
        }

        if(n == 2)
        {
            // 其实就是twoSum代码框架
            int left = start, right = sz - 1;

            while(left < right)
            {
                int leftNum = nums[left], rightNum = nums[right];
                int sum = leftNum + rightNum;

                if(sum < target)
                {
                    left++;
                }
                else if(sum > target)
                {
                    right--;
                }
                else if(sum == target)
                {
                    res.push_back({leftNum, rightNum});

                    while(left < right && leftNum == nums[left])
                        left++;
                    while(left < right && rightNum == nums[right])
                        right--;
                }
            }
        }
        else
        {
            // n > 2的情况,递归计算(n-1)Sum的结果
            for(int i = start; i < sz; i++)
            {
                // 也就是说,计算三数之和,先递归计算两数之和
                vector<vector<int>> twoNums = nSum(nums, n - 1, i + 1, target - nums[i]);

                // 将两数之和的结果加上当前的元素就是题目所求的三数之和
                for(auto twoNum : twoNums)  
                {
                    // (n-1)Sum 加上 nums[i] 就是 nSum
                    twoNum.push_back(nums[i]);
                    res.push_back(twoNum);
                }

                // 防止第一个元素重复
                while(i < sz - 1 && nums[i] == nums[i + 1])
                {
                    i++;
                }
            }
        }

        return res;
    }
};

代码虽然看起来比较长,但是只要理解了就很简单,因为n==2时就是twoSum的双指针解法,n > 2时就是穷举第一个数字,然后递归计算(n-1)Sum,组长答案。

还有一点需要注意的是,类似 twoSum3Sum 的结果也可能重复,比如输入是 nums = [1,1,1,2,3], target = 6,结果就会重复。

关键点在于,不能让第一个数重复,至于后面的两个数,我们复用的 twoSum 函数会保证它们不重复。所以代码中必须用一个 while 循环来保证 nSum 中第一个元素不重复。

三、四数之和

原题链接:四数之和

在这里插入图片描述

解题思路:

没啥好说的了,理解了上面的三数之和这道题自然就没有问题。

只有一点需要注意,就是target类型需要定义为long long 防止溢出。因为这道题说了 nums[i]target 的取值都是 [-10^9, 10^9]int 类型的话会造成溢出。

代码详解:

class Solution {
public:
    vector<vector<int>> fourSum(vector<int>& nums, int target) 
    {
        // 先对数组进行排序
        sort(nums.begin(), nums.end());

        return nSum(nums, 4, 0, target);
    }

    vector<vector<int>> nSum(vector<int>& nums, int n, int start, long target)
    {
        int sz = nums.size();

        vector<vector<int>> res;

        if(n < 2 || n > sz)
            return res;

        if(n == 2)
        {
            int left = start, right = sz - 1;

            while(left < right)
            {
                int leftNum = nums[left], rightNum = nums[right];
                int sum = leftNum + rightNum;

                if(sum < target)
                {
                    left++;
                }
                else if(sum > target)
                {
                    right--;
                }
                else if(sum == target)
                {
                    res.push_back({leftNum, rightNum});

                    while(left < right && leftNum == nums[left])
                        left++;
                    while(left < right && rightNum == nums[right])
                        right--;
                }
            }
        }
        else
        {
            // n > 2
            for(int i = start; i < sz; i++)
            {
                vector<vector<int>> threeNums = nSum(nums, n - 1, i + 1, target - nums[i]);

                for(auto threeNum : threeNums)
                {
                    threeNum.push_back(nums[i]);
                    res.push_back(threeNum);
                }

                // 防止第一个元素重复
                while(i < sz - 1 && nums[i] == nums[i + 1])
                {
                    i++;
                }
            }
        }

        return res;
    }
};
遇见安然遇见你,不负代码不负卿。
谢谢老铁的时间,咱们下篇再见~

在这里插入图片描述

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

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

相关文章

Flink集群的搭建

1、Flink独立集群模式 1、首先Flink的独立集群模式是不依赖于Hadoop集群。 2、上传压缩包&#xff0c;配置环境&#xff1a; 1、解压&#xff1a; tar -zxvf flink-1.15.2-bin-scala_2.12.tgz2、配置环境变量&#xff1a;vim /etc/profileexport FLINK_HOME/usr/local/soft/fl…

C++深度优先搜索(DFS)算法的应用:树中可以形成回文的路径数

本文涉及知识点 深度优先搜索(DFS) 状态压缩 题目 给你一棵 树&#xff08;即&#xff0c;一个连通、无向且无环的图&#xff09;&#xff0c;根 节点为 0 &#xff0c;由编号从 0 到 n - 1 的 n 个节点组成。这棵树用一个长度为 n 、下标从 0 开始的数组 parent 表示&#…

3.2-Docker Image概述

常用docker命令&#xff1a; 查看docker image有哪些 docker image ls Image的获取方式

亚马逊云科技产品测评』活动征文|通过使用Amazon Neptune来预测电影类型初体验

文章目录 福利来袭Amazon Neptune什么是图数据库为什么要使用图数据库什么是Amazon NeptuneNeptune 的特点 快速入门环境搭建notebook 图神经网络快速构建加载数据配置端点Gremlin 查询清理 删除环境S3 存储桶删除 授权声明&#xff1a;本篇文章授权活动官方亚马逊云科技文章转…

jstat虚拟机统计信息监控工具

jstat虚拟机统计信息监控工具 1、jstat&#xff08;JVM Statistics Monitorning Tool&#xff09; 用于监控虚拟机各种运行状态信息的命令行工具。 它可以显示本地或远程虚拟机进程中的类装载、内存、垃圾收集、JIT编译等运行数据&#xff0c;它是运行期定位虚拟机 性能问题…

Lenovo联想IdeaPad 15 ALC7(82R4)2022款笔记本原装出厂Windows11预装系统镜像

下载链接&#xff1a;https://pan.baidu.com/s/1Pr2G42Sz1LKzgF8OcEBkPQ?pwd5ms2 提取码&#xff1a;5ms2 系统自带所有驱动、出厂主题壁纸、系统属性专属LOGO标志、Office办公软件、联想电脑管家等预装程序 所需要工具&#xff1a;16G或以上的U盘 文件格式&#xff1a;…

centos7.9 postgresql 16.0 源码安装部署

postgresql 16.0 源码安装部署 环境准备 系统主机名IP地址centos7.9postgres192.168.200.56 软件准备 postgresql-16.0.tar.gz https://ftp.postgresql.org/pub/source/v16.0/postgresql-16.0.tar.gz依赖安装 yum -y install systemd-devel readline readline-devel zlib-devel…

探秘Python闭包与作用域

文章目录 闭包的定义与作用LEGB规则nonlocal与global关键字在Python的世界里,理解闭包(Closure)和作用域(Scope)是提升编程技巧和深度的一大步。这篇文章将带你深入了解闭包的神秘面纱,掌握LEGB规则,并使用nonlocal与global关键字来巧妙控制变量作用域。 闭包的定义与作…

MySQL中的刷脏机制详解

名词解释 脏页&#xff1a;当内存数据页跟磁盘数据页内容不一致的时候&#xff0c;我们称这个内存页为“脏页”。 干净页&#xff1a;内存数据写入到磁盘后&#xff0c;内存和磁盘上的数据页的内容就一致了&#xff0c;称为“干净页”。 LSN&#xff1a;称为日志的逻辑序列号(l…

万物社用户运营工具:无代码开发下的电商平台和CRM集成

简介&#xff1a;万物社与集简云的引领式连接 万物社&#xff0c;隶属于厦门头号云信息科技有限公司&#xff0c;是一家专注于互联网和相关服务的企业。在日常的业务运营中&#xff0c;万物社通过与集简云的无代码集成&#xff0c;实现了业务流程的自动化和智能化&#xff0c;…

学习笔记4——JVM运行时数据区梳理

学习笔记系列开头惯例发布一些寻亲消息 链接&#xff1a;https://baobeihuijia.com/bbhj/contents/3/192489.html 类装载器classLoader&#xff1a; 将本地的字节码文件.class 加载到内存方法区中成为元数据模板&#xff08;两个class对象是否为同一个类要求&#xff1a;完整…

对话句子互动创始人李佳芮 | AIGC结合私域运营影响不可估量

“ 创业最核心的就是耐心” 口述 | 李佳芮 整理 | 小白&云舒 出品&#xff5c;极新 极新请文心一言分析了私域流量运营和chatbot当下的发展背景&#xff0c;它给出了以下答案&#xff1a; 1. 移动设备普及和网络速度提升&#xff1a;随着智能手机和移动互联网的普及&…

UE5.3实现1秒12帧风格的动画抽帧效果

现今一些卡通风格游戏会刻意模仿早期动画1秒12帧的播放效果&#xff0c;以营造较强的风格化体验&#xff0c;博主在UE5中实现了一下&#xff08;左侧正常动画&#xff0c;右侧抽帧动画&#xff09;&#xff1a; 我们可以通过在UE中对导入设置进行一些修改&#xff0c;达到不改…

Android UI 开发·界面布局开发·案例分析

目录 ​编辑 1. 线性布局&#xff08;LinearLayout&#xff09; 2. 相对布局&#xff08;RelativeLayout&#xff09; 3. 表格布局&#xff08;TableLayout&#xff09; 4. 帧布局&#xff08;FrameLayout&#xff09; 5. 网格布局&#xff08;GridLayout&#xff0…

AI:68-基于深度学习的身份证号码识别

🚀 本文选自专栏:AI领域专栏 从基础到实践,深入了解算法、案例和最新趋势。无论你是初学者还是经验丰富的数据科学家,通过案例和项目实践,掌握核心概念和实用技能。每篇案例都包含代码实例,详细讲解供大家学习。 📌📌📌在这个漫长的过程,中途遇到了不少问题,但是…

C语言编写一个程序采集招聘信息

因为在这里无法详细解释每行代码和步骤。但是&#xff0c;我可以给大家一个使用Python和requests库编写的简单爬虫程序的例子&#xff0c;它可以从网站上获取招聘信息。你可以根据这个例子&#xff0c;将其改写为使用C语言编写的爬虫程序。 import requests# 指定爬虫IP信息 pr…

二叉树的OJ题——C++

一、根据二叉树创建字符串 题目链接&#xff1a; 606. 根据二叉树创建字符串 - 力扣&#xff08;LeetCode&#xff09; 题目描述&#xff1a; 前序遍历二叉树&#xff0c;并且将结果存入到一个string中&#xff0c;并且要使用括号去分割和表示每个节点和子树之间的关系&…

大漠插件(二、Qt使用插件时注意事项)

本章目的 在上篇已经注册完毕大漠&#xff0c;那么怎么使用大漠来制作脚本&#xff0c;我选择了我最熟悉的Qt来开发&#xff0c;毕竟只是小软件&#xff0c;用脚本或者c都差不了多少。本章就是开发途中的一些坑。 本人开发环境是 win11 64、Qt 5.15.2安装了5.10.0的msvc2015 32…

接口测试工具的实验,Postman、Swagger、knife4j(黑马头条)

一、Postman 最常用的接口测试软件&#xff0c;需要注意点&#xff1a;在进行post请求时&#xff0c;需要选择JSON形式发送 输入JSON字符串&#xff0c;比如&#xff1a; {"maxBehotTime": "2021-04-19 00:19:09","minBehotTime": "2021-…

支持存档的书签服务LinkWarden

什么是 LinkWarden &#xff1f; Linkwarden 是一个自托管、开源协作书签管理器&#xff0c;用于收集、组织和存档网页。目标是将您在网络上找到的有用网页和文章组织到一个地方&#xff0c;并且由于有用的网页可能会消失&#xff08;参见链接失效的必然性&#xff09;&#xf…