代码随想录day4 | 24. 两两交换链表中的节点 19.删除链表的倒数第N个节点 02.07.链表相交 142.环形链表II

news2025/1/11 21:38:37

文章目录

    • 一、两两交换链表中的节点
    • 二、删除链表的倒数第N个节点
    • 三、链表相交
    • 四、环形链表

24. 两两交换链表中的节点
19.删除链表的倒数第N个节点
面试题 02.07. 链表相交
142.环形链表II

一、两两交换链表中的节点

两两交换链表中的节点

注意是两两交换,采用虚拟头结点的方法,因为每次操作单链表都需要找到前一个指针。奇数节点数,最后一个不变;偶数节点数,就是正常的两两交换。
操作节点cur一定要指向要交换的两个节点的前一个节点!

class Solution
{
public:
    ListNode *swapPairs(ListNode *head)
    {
        ListNode *dummy = new ListNode(-1);
        dummy->next = head;
        ListNode *cur = dummy;
        while (cur->next != nullptr && cur->next->next != nullptr)
        {
            // 记录两个临时节点,方便后续更新操作
            ListNode *tmp = cur->next;
            ListNode *tmp1 = cur->next->next->next;

            cur->next = cur->next->next; // 步骤1
            cur->next->next = tmp;  // 步骤2
            tmp->next = tmp1;  // 步骤3

            cur = cur->next->next; // cur后移两位
        }
        return dummy->next;
    }
};

整体流程:
在这里插入图片描述

在这里插入图片描述

ListNode * cur =dummy; // 为啥初始化为dummy?

因为这样才能操作刚刚开始的节点。

while(?) 循环结束条件是啥?

这个要看节点数n的奇偶
在这里插入图片描述

3、用&&而不是||

使用 && 而不是 || 是为了确保链表中至少有两个节点才能进行节点交换。
在这里插入图片描述

4、不能交换两个cur->next != nullptr && cur->next->next != nullptr 顺序

如果将它们的顺序互换,即 cur->next->next != nullptr && cur->next != nullptr,会导致在访问 cur->next->next 之前访问 cur->next,这可能会产生空指针异常(Null Pointer Exception),因为如果 cur->next 是 nullptr,那么访问 cur->next->next 就会出错。

5、为啥定义临时变量tmp?
因为后续改变指向后,无法再找到该节点的前驱和后继了。
趁着还没有赋值操作步骤1前,把cur->next 保存下来,tmp=cur->next,tmp1=cur->next->next->next;
在这里插入图片描述

二、删除链表的倒数第N个节点

19.删除链表的倒数第N个节点

删除的思路都是,直接指向next的next,就相当于删除了。
还是采用虚拟头结点,方便我们删除,不需要特判是不是头删!采用统一的方式进行删除!

class Solution
{
public:
    ListNode *swapPairs(ListNode *head)
    {
        ListNode *dummy = new ListNode(-1);
        dummy->next = head;
        ListNode *cur = dummy;
        while (cur->next != nullptr && cur->next->next != nullptr)
        {
            // 记录两个临时节点,方便后续更新操作
            ListNode *tmp = cur->next;
            ListNode *tmp1 = cur->next->next->next;

            cur->next = cur->next->next;
            cur->next->next = tmp;
            tmp->next = tmp1;

            cur = cur->next->next;
        }
        return dummy->next;
    }
};
/**********************************分割线*******************************************/
// 双指针删除 倒数第n个节点
class Solution
{
public:
    ListNode *removeNthFromEnd(ListNode *head, int n)
    {
        ListNode *dummy = new ListNode(-1);
        dummy->next = head;
        ListNode *fast = dummy;
        ListNode *slow = dummy;

        // 快指针先走n步  , n--就是n步
        while (n-- && fast != nullptr)
        {
            fast = fast->next;
        }
        fast = fast->next;// 多走一步

        /**************************/
        // 快慢指针同时走
        while (fast != nullptr)
        {
            fast = fast->next;
            slow = slow->next;
        }
        slow->next = slow->next->next; // 删除
        return dummy->next;
    }
};

如何找到倒数第n个节点?采用快慢指针的方法。

fast首先走n + 1步 ,为什么是n+1呢

因为只有这样同时移动的时候slow才能指向删除节点的上一个节点(方便做删除操作)
在这里插入图片描述

在这里插入图片描述

三、链表相交

面试题 02.07. 链表相交
双指针解法
定义两个指针,a,b,分别从各自的链表头部开始,同时走;a指针走到空了,就把a更新到链表b,b指针也是如此,直到两者相遇。

class Solution
{
public:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB)
    {
        ListNode *a = headA, *b = headB;
        while (a != b)
        {
            if (a)
                a = a->next;
            else
                a = headB;

            if (b)
                b = b->next;
            else
                b = headA;
        }
        return a;
    }
};

四、环形链表

142.环形链表II

class Solution
{
public:
    ListNode *detectCycle(ListNode *head)
    {
        ListNode *fast = head, *slow = head;
        while (fast != nullptr && fast->next != nullptr) // 为啥要判断fast->next?
        {
            fast = fast->next->next;
            slow = slow->next;
            // 快慢指针相遇,此时从head 和 meet相遇点,同时新定义节点开始查找直至相遇
            if (fast == slow)
            {
                ListNode *newhead = head;
                ListNode *meet = fast;
                while (newhead != meet)
                {
                    newhead = newhead->next;
                    meet = meet->next;
                }
                return newhead;
            }
        }
        return nullptr;
    }
};

1、为何想到用快慢指针来判断有没有环?

假设没有环,那么fast和slow是不可能相遇的;有环的话,那么fast会先进入环中,在里面打转,slow后进入,两者总能够相遇。

在这里插入图片描述

2、为何fast一次走两步,slow一次走一步?

进入环后,相当于是fast在以相对速度为1步来追击slow,那么他们必然相遇;如果是其他步数的话,他们两者在环中可能会刚好错过!

3、如何找到入环点?

代码随想录完整证明
在这里插入图片描述
他们相遇时:
slow指针走过的路程为: x + y
fast指针走过的路程为:x + y + n (y + z)
速度:slow为1,fast为2
时间是相等的
所以得到等式:
在这里插入图片描述
整理得到:x = (n - 1) (y + z) + z,当n=1时,x=z

结论:这就意味着,从头结点出发一个指针,从相遇节点也出发一个指针,这两个指针每次只走一个节点, 那么当这两个指针相遇的时候就是环形入口的节点。

4、while(fast!=nullptr && fast ->next!=nullptr) // 为啥要判断fast->next?

因为fast一次条两步!

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

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

相关文章

Global symbol “%data“ requires explicit package name

Global symbol “%data” requires explicit package name 如图编写demo的时候出现了如图的问题,在网上查找到的原因是: 一,使用use strict; ,修改其他代码,如下: 1,首先,检查你是不…

静态库和动态库的区别与优缺点

文章目录 静态库与动态库的区别动态库与静态库的优缺点 静态库与动态库的区别 静态库直接打包链接到可执行程序 动态库将不会链接到可执行文件 ,执行文件运行时需要动态加载 动态库 ,所以需要提前知道动态库路径,需要将路径保存到环境变量或…

亲测:2核4G云服务器性能测试能安装多少个网站?

2核4G服务器能安装多少个网站?2核4g配置能承载多少个网站?一台2核4G服务器可以安装多少个网站?阿腾云2核4G5M带宽服务器目前安装了14个网站,从技术角度是没有限制的,只要云服务器性能够用,想安装几个网站就…

【前端知识】React 基础巩固(二十六)——Portals 的使用

React 基础巩固(二十六)——Portals 的使用 Portals 通常&#xff0c;组件会渲染到 root 节点下。可使用 Portals 将组件渲染至其他节点。 添加 id 为 more、modal 的 div 元素 <div id"root"></div> <div id"more"></div> &l…

apache 工作模式 NPM Multi-processing module

Apache 三种工作模式 prefork work event 监控线程相当于 项目组长&#xff0c;面对客户&#xff0c;根据任务自行调度 团队成员。 Prefork模式为Apache默认使用的工作模式&#xff0c;在该模式下&#xff0c;采用的机制是预派生子进程的方式&#xff0c;使用单独的进程来…

设计模式之单例

文章目录 前言一、单例模式的基本定义二、懒汉式单例三、饿汉式单例四、懒汉式单例存在的线程安全问题总结 前言 单例模式是比较经典的设计模式&#xff0c;本篇我们将结合实际案例说明单例模式的实现原理&#xff0c;以及存在的问题。 一、单例模式的基本定义 单例模式&…

【前端知识】React 基础巩固(二十八)——StrictMode

React 基础巩固(二十八)——StrictMode StrictMode StrictMode 是一个用来突出显示应用程序中潜在问题的工具 与 Fragment 一样&#xff0c;StrictMode 不会渲染任何可见的 UI为后代出发额外的检测和警告严格模式检查仅在开发模式下运行&#xff0c;不影响生产构建 严格模式检…

Google DeepMind掌舵人Demis Hassabis专访:合并后「超级单元」内幕,以及如何开展下一代模型研究...

导读 自2023年以来&#xff0c;Google旗下两大顶级人工智能研究团队 DeepMind 与 Google Brain 被不断推上风口浪尖&#xff0c;如何应对OpenAI与微软强强联合&#xff1f;如何组织不被看好的 DeepMind 与 Google Brain 团队合并&#xff1f;Google和OpenAI都没有护城河吗&…

单片机第一季:零基础8——蜂鸣器

蜂鸣器是一种一体化结构的电子讯响器&#xff0c;采用直流电压供电&#xff0c;广泛应用于计算机、打印机、复印机、报警器、电子玩具、汽车电子设备、电话机、定时器等电子产品中作发声器件。 蜂鸣器主要分为压电式蜂鸣器和电磁式蜂鸣器两种类型。 想要压电式蜂鸣器发声&…

使用Gataway设置全局过滤器

说明&#xff1a;Gateway网关是所有微服务统一的入口&#xff0c;通过网关技术&#xff0c;可以实现请求转发、权限控制和限流。全局过滤器是Gateway请求转发、权限控制的一个体现。比如对所有请求进行过滤&#xff0c;判断请求是否携带有效的token&#xff0c;对请求放行或者拦…

Java 装箱拆箱原理 包装类型缓存池

JAVA中的基本数据类型 byteshortintlongfloatdoublebooleanchar 为了让上述基本数据类型可以转为对象&#xff0c;Java在1.5推出了一系列包装类&#xff0c;基本类和包装类互相转换的过程&#xff0c;称为装箱和拆箱 缓存池 缓存池也叫常量池。它是事先存储一些常量数据用以…

reverse

b,a,0x7d65都能轻易的转成字符串&#xff0c;就是这个负数需要看汇编&#xff0c;在汇编层面是加了两个数实现的减法效果。 a[galf,saE{,er_y,3r0v,}e] for each in a:print(each[::-1],end) # flag{Easy_rev0r3e}

华为OD机试真题 Java 实现【路灯照明问题】【2022Q4 100分】,感谢fly晨发现这个问题,并提供更优质的算法

目录 专栏导读一、题目描述二、输入描述三、输出描述四、解题思路特别鸣谢&#xff1a;感谢fly晨发现这个问题&#xff0c;并提供更优质的算法。解题思路如下&#xff1a; 五、Java算法源码六、效果展示1、输入2、输出3、思路 专栏导读 本专栏收录于《华为OD机试&#xff08;J…

【USRP X310】如何查找USRP X310的 LabVIEW 案例 Demo

环境 Windows 10LabVIEWUSRP 支持的硬件 USRP-2900USRP-2901USRP-2920USRP-2921USRP-2922USRP-2930USRP-2932USRP-2940 40 MHzUSRP-2940 120 MHzUSRP-2942 40 MHzUSRP-2942 120 MHzUSRP-2943 40 MHzUSRP-2943 120 MHzUSRP-2944USRP-2945USRP-2950 40 MHzUSRP-2950 120 MHzUS…

轻奢分销商城pc/h5小程序开源版开发

轻奢分销商城pc/h5小程序开源版开发 商城系统&#xff1b;包含小程序商城、H5商城、公众号商城、PC商城、App&#xff0c;支持秒杀、优惠券、积分、分销、会员等级。 功能列表&#xff1a; 用户注册和登录&#xff1a;用户可以通过手机号码或第三方登录方式注册和登录账号。…

RPM包详解以及如何制作RPM包

RPM包简介 RPM 全名 RedHat Package Managerment&#xff0c;一开始只是Red Hat Linux的软件包格式&#xff0c;它最初的名字为Red Hat软件包管理程序。从那以后&#xff0c;RPM逐渐称为许多其他Linux发行版所接受的一种软件包格式。Linux标准化规范将RPM作为其官方软件包格式…

gdb调试产生code文件以及遇到的“file format not recognized”问题解决

一、调试 core 文件的配置 1.1 开启core文件 —— ulimit 有时候&#xff0c;服务器程序运行一段时间后会突然崩溃&#xff0c;这并不是我们希望看到的&#xff0c;需要 解决这个问题。只要程序在崩溃的时候有 core 文件产生&#xff0c;就可以使用这个 core 文件 来定位崩溃…

数据结构(王道)——线性表的存储结构之双链表

双链表和单链表的对比 一、从无到有创建一个双链表及其基础操作 带头结点 的双链表 初始化 双链表的后插操作&#xff1a; 从后插可以拓展到按位序插入&#xff0c;和前插操作。 因为双链表的特性&#xff0c;如果按位序插入或者前插操作&#xff0c;只需要找到插入的位置的前…

传统企业的数据管理:如何高效管理分散的数据?

在数字时代&#xff0c;数据管理已经成为一个重要的议题。大多数企业已经意识到数据是他们业务运营的重要组成部分。然而&#xff0c;许多传统企业还在使用过时的数据管理模式&#xff0c;这些模式并不具备高效管理分散数据的能力。随着企业的膨胀和数据规模的逐渐增加&#xf…