算法进阶:双指针(一)c++leetcode例题

news2025/1/17 15:36:51

82. 删除排序链表的重复元素

力扣传送:
https://leetcode.cn/problems/remove-duplicates-from-sorted-list-ii/description/

给一个排好序的链表,删除把链表中出现的所有的重复的项:
1 2 2 3 3 3 4 5 -----> 1 4 5

这道题有很多种解法,我第一眼看到这题的时候,想到了 哈希表。
利用哈希表来记录出现的重复的节点元素,接着在遍历链表,等到这个节点出现在了哈希表中,则while 循环一直跳过这个节点,直到跳到下一个不同的元素为止。

在这里插入图片描述


哈希表的做法:

  1. 首先遍历一边链表,记录下哈希表的数据
  2. 然后再遍历一遍链表,当哈希表记录的此节点处的值出现的次数大于一次时,一直while找到直到下一个等于一次的节点位置,然后连接前一个节点位置,这样就做到了跳过了出现次数大于二的节点的中间节点。
  3. 双指针分别连接链表节点。
class Solution {
public:
    ListNode* deleteDuplicates(ListNode* head) {
        unordered_map<int,int> m;
        ListNode* temp=head;
        while (temp)
        {
  			//首先初始化哈希表,记录下每个节点的值出现的次数
            m[temp->val]++;
            temp=temp->next;
        }
        ListNode* pDummy=new ListNode{-101,head};
        ListNode* slow=pDummy;
        ListNode* fast=head;
        while (fast && fast->next)
        {
            if (m[fast->val]>1)
            {
                ListNode* pTemp=fast->next;
                while (fast && m[fast->val]>1)
                {
                    fast=fast->next;
                }
                slow->next=fast;
            }
            else
            {
                slow=slow->next;
                fast=fast->next;
            }
        }
        return pDummy->next;
    }
};

哈希表的缺点:
我们的哈希表的空间随着链表的增大而增大,空间复杂度达到了O(N)。

但其实我们没必要使用哈希表。
我们知道用哈希表来记录出现次数大于一次的节点,那我们能不能直接用双指针的快指针来记录呢? 只需要合适的边界检测,就可以记录下其快指针的 当前节点元素和 下一个节点元素的值,判断他们是否相等即可。

实现:

  1. 哑节点方便返回链表(pDummy)
  2. 慢指针在前,快指针在后,快指针移动一步,每次快指针都判断它当前的值和后面的值想不想等,如果相等,则快指针进入while循环,一直跳过。
  3. 等到处理完毕后, 慢指针前进一步,等于快指针;快指针再前进一步,等于next
class Solution {
public:
    ListNode* deleteDuplicates(ListNode* head) {
        ListNode* pDummy=new ListNode{-101,head};
        ListNode* slow=pDummy;
        ListNode* fast=head;
        while (fast && fast->next)
        {
            int val=fast->val;
            if (val==fast->next->val)
            {
                while (fast && fast->val==val)
                {
                    fast=fast->next;
                }
                slow->next=fast;
            }
            else
            {
                slow=fast;
                fast=fast->next;
            }
        }
        return pDummy->next;
    }
};

时间复杂度:O(N)只遍历链表一遍,空间复杂度:O(1)


15. 三数之和

力扣传送门:
https://leetcode.cn/problems/3sum/description/?envType=study-plan&id=suan-fa-ji-chu&plan=algorithms&plan_progress=y00ve32

找到数组中是否包含三个元素使得 nums[i]+nums[j]+nums[z] =0

这道题目一看看出就可以使用暴力搜索的做法:首先给数组排序,然后创建一个三重循环,每重循环遍历每一个元素;注意,我们的元素可能会出现重复项,因此我们要跳过两重循环中可能遍历到的重复的项:

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        vector<vector<int>> res;
        int n=nums.size();
        sort(nums.begin(),nums.end());
        for (int i=0;i<n;i++)
        {
            if (i==0 || nums[i]!=nums[i-1])
            {
                for (int j=i+1;j<n;j++)
                {
                    if (j==i+1 || nums[j]!=nums[j-1])
                    {
                        for (int z=j+1;z<n;z++)
                        {
                            if (z==j+1 || nums[z]!=nums[z-1])
                            {
                                 if (nums[i]+nums[j]+nums[z]==0)
                                {
                                    res.push_back({nums[i],nums[j],nums[z]});
                                }
                            }
                        }
                    }
                }
            }
        }
        return res;
    }
};

我们的时间复杂度达到了O(N^3) 这对于最长数据量3000来说是很容易超时的,因为 3000 * 3000 * 3000 相乘后达到了27,000,000,000,即这就是三重循环的最坏的情况,因此暴力搜索的方法无法通过。


我们引入一个新方法:

利用双指针降低维数: 利用双指针把三重循环降低为二重循环。

  1. 当我们确定第一个数字之后,如果第二个数字增大,则第三个数字必定减小。
  2. 令第二个数字和第三个数字同时寻找。
  3. 第二个数字起始位于第二重循环的开头,第三个数字起始位于第二重循环的末尾。

我们就可以保持第二重循环不变,而将第三重循环变成一个从数组最右端开始向左移动的指针
在这里插入图片描述

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        int n=nums.size();
        vector<vector<int>> res;
        sort(nums.begin(),nums.end());
        for (int i=0;i<n;i++)
        {
            //当前项和前一项相同,则当前数字已经被刚才用过了,则直接跳过这个数字
            if (i>0 && nums[i]==nums[i-1])
            {
                continue;
            }
            //第二重循环同时遍历 j 和 z
            int z=n-1;  //初始化z的位置,z从后往前
            for (int j=i+1;j<n;j++)   //j从前往后
            {
                //同理,跳过重复的数字
                if (j>i+1 && nums[j]==nums[j-1])
                {
                    continue;
                }
                //同时需要保证j<z:因为j作为左指针一定要在z右指针的左边
                while (j<z && nums[i]+nums[j]+nums[z]>0)
                {
                    //因为序列从小到大排序,当前的结果大于0,则减小z,寻找合适的位置
                    --z;
                }
                //如果j 和 z相遇,则表示无论j再往后,z再往前,他们都不可能再有结果了(和为0),因为j再往后遍历的数字一定和z之前的一个数字相同; z也是,一定和j之前的一个数字相同,我们已经遍历过了,所以这种情况直接退出 
                if (j==z)
                {
                    break;
                }
                if (nums[i]+nums[j]+nums[z]==0)
                {
                    res.push_back({nums[i],nums[j],nums[z]});
                }
            }
        }
        return res;
    }
};

总结:

当我们需要枚举数组中的两个元素时,如果我们发现随着第一个元素的递增,第二个元素是递减的,那么就可以使用双指针的方法,将枚举的时间复杂度从 O(N^2)减少至 O(N)

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

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

相关文章

Unity3D——第一人称FPS生存游戏(resident zombies)

游戏源文件和游戏试玩程序:链接&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1Ln2tFizqEO_uEoQhuxvgrQ?pwdl6w0 提取码&#xff1a;l6w0 游戏思路前身搭建: 用一些正方体和胶囊做出来的基础场景&#xff0c; 人物设计:红色的胶囊体是敌人&#xff0c;手持枪械是…

一文说透BIO以及非阻塞IO

目录1. 一次I/O到底经历了什么2. 什么是Socket3. 阻塞I/O&#xff08;Blocking I/O&#xff0c;BIO&#xff09;3.1. 客户端的socket流程3.1.1. 何为socket&#xff1f;3.1.2. 何为连接&#xff1f;3.2. 服务端的socket流程3.2.1. 创建socket3.2.2. 绑定端口号3.2.3. listen()的…

项目实战之旅游网(六)认证与授权

一.编写相关页面 在本项目中&#xff0c;我们使用Spring Security 进行认证和授权&#xff0c;首先我们先编写相关页面。 1.编写登录页面admin_ login.html 2.编写登录失败页面admin_ fail.html 3.编写权限不足页面no_ permission.html 略过。知道其功能即可。 二.编写配置…

数字孪生城市可视化大屏设计,智慧楼宇开源项目

纵观城市发展历史&#xff0c;技术的革命必然会带动城市内部的变革。当前&#xff0c;以数字孪生为代表的前言信息技术飞速发展&#xff0c;必然会使社会对数字城市的深度和广度有着更为清晰的认知。加快构建数字孪生城市管理平台&#xff0c;通过三维可视化大屏直观展示&#…

图表控件LightningChart.NET 系列教程(六):许可证管理介绍(中)

LightningChart.NET SDK 是一款高性能数据可视化插件工具&#xff0c;由数据可视化软件组件和工具类组成&#xff0c;可支持基于 Windows 的用户界面框架&#xff08;Windows Presentation Foundation&#xff09;、Windows 通用应用平台&#xff08;Universal Windows Platfor…

Apache ShardingSphere-Proxy <5.3.0 存在身份认证绕过漏洞

漏洞描述 Apache ShardingSphere 是一款分布式的数据库生态系统&#xff0c;ShardingSphere-Proxy 是支持 MySQL、PostgreSQL 和 openGauss 协议的数据库代理模块。 ShardingSphere-Proxy 5.3.0 之前的版本中在使用 MySQL 作为后端数据库时&#xff0c;在客户端认证失败后没有…

Linux进程状态与优先级

Ⅰ. OS进程状态的概念 进程状态反映进程执行过程的变化。这些状态随着进程的执行和外界条件的变化而转换。 在三态模型中&#xff0c;进程状态分为 运行态&#xff0c;就绪态&#xff0c;阻塞态。 在五态模型中&#xff0c;进程状态分为 新建态、终止态&#xff0c;运行态&a…

服务雪崩预防Sentinel

服务雪崩效应 在分布式系统中,由于网络原因或自身的原因,服务一般无法保证 100% 可用。如果一个服务出现了 问题&#xff0c;调用这个服务就会出现线程阻塞的情况&#xff0c;此时若有大量的请求涌入&#xff0c;就会出现多条线程阻塞等 待&#xff0c;进而导致服务瘫痪。 由于…

【HTML+CSS+JavaScript】实现鼠标点击烟花效果

文章目录【HTMLCSSJavaScript】实现鼠标点击烟花效果&#xff08;爆炸型、心型、圆形&#xff09;一. 效果图二. 鼠标点击烟花效果 - 心型实现代码(1) HTML部分代码(2) CSS部分代码(3) 内部的JavaScript部分代码三. 鼠标点击烟花效果 - 圆型实现代码(1) HTML部分代码(2) CSS部分…

【mysql】优化系列文章之一-索引

mysql优化系列 不是教程&#xff0c;不是官方文档&#xff0c;而是自己实战的点滴记录&#xff0c;不一定适合新手和系统学习者 第一章 mysql索引 文章目录mysql优化系列前言1、Mysql索引2、B Tree2.1.特点2.2. 结构分解2.3. 例题分析2.4. 验证索引2.5.索引插入耗时3. MySQL 中…

Oracle数据库同步复制工具Beedup产品功能(一)

1、全量复制 Beedup全量复制功能通过遍历比对主从库用户模式及其下包含的各类对象来保证主从库的相关对象一致性。 支持角色、用户、架构、登录用户、表 (列定义 主外键 索引)、视图存储过程、函数、触发器、类型、类型体、包、包体、序列、同义词、数据库链接等对象复制全量…

技术干货 | 人大金仓KFS基于分区索引的分片入库技术解析

在之前的文章《技术干货&#xff1a;人大金仓KFS精准过滤和分片并行入库技术解析》中&#xff0c;KFS利用分片并行入库技术&#xff0c;解决了某金融POC数据同步项目中数据入库持续积压问题。经过优化后&#xff0c;在200并发的压测场景中&#xff0c;整体同步性能指标从压测30…

基于BP神经网络、RBF神经网络以及PSO优化的RBF神经网络进行数据的预测(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️❤️&#x1f4a5;&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑…

IT30--IT之职能部门(3年之约已满)

IT作为一个职能部门&#xff0c;肯定跟业务不同&#xff0c;具体有哪些事情要做&#xff0c;我们一一道来。 年终规划的二三事 组织体系规划及梳理 预算编制 今天我们先来说说组织体系规划。 1 从愿景使命价值观到行动及思考方法 无论是企业还是部门&#xff0c;软文件建设肯…

【信息检索与数据挖掘期末笔记】(六)Link Analysis

Web图 将Web当做有向图 节点&#xff1a;网页 边&#xff1a;超链接 PageRank 不同网页的重要性是不同的&#xff0c;在web-graph中&#xff0c;节点之间的连接性有巨大的差异。我们根据链接结果来对网页进行排序 想法&#xff1a;用链接来投票 如果有更多的链接指向一个网…

Linux Shell 编程,Shell 变量详解

Linux Shell 编程&#xff0c;Shell 变量详解1.第一个shell脚本2.Shell 变量初探3.位置参数变量4.预定义变量1.第一个shell脚本 打开文本编辑器(可以使用 vi/vim 命令来创建文件)&#xff0c;新建一个文件 hello.sh&#xff0c;扩展名为 sh&#xff08;sh代表shell&#xff09…

【算法】面试题 - 链表(附讲解视频)

链表相关面试题876. 链表的中间结点206. 反转链表86. 分隔链表160. 相交链表141. 环形链表问题&#xff1a;快慢指针为什么一定会相遇142. 环形链表 II问题&#xff1a;如何确认入口237. 删除链表中的节点19. 删除链表的倒数第 N 个结点21. 合并两个有序链表23. 合并K个升序链表…

【记忆增强深度条件展开网络】

Memory-augmented Deep Conditional Unfolding Network for Pan-sharpening &#xff08;面向全色锐化的记忆增强深度条件展开网络&#xff09; 全色锐化旨在为遥感系统获取高分辨率的多光谱图像&#xff0c;基于深度学习的方法已经取得了显著的成功。然而&#xff0c;大多数…

使用iServer rest api如何实现构建巷道效果

作者&#xff1a;刘大 背景 在实际生产环境中&#xff0c;特别是在采矿&#xff0c;公路建设项目上&#xff0c;我们往往会接触下图所示的巷道&#xff0c;那么在Web端如何快速通过线数据构建巷道模型呢&#xff1f;下面我们来详细说下 使用方式 第一步&#xff1a; 在iServe…

反垃圾邮件系统|基于Springboot+vue 实现反垃圾邮件系统

作者主页&#xff1a;编程指南针 作者简介&#xff1a;Java领域优质创作者、CSDN博客专家 、掘金特邀作者、多年架构师设计经验、腾讯课堂常驻讲师 主要内容&#xff1a;Java项目、毕业设计、简历模板、学习资料、面试题库、技术互助 收藏点赞不迷路 关注作者有好处 文末获取源…