环形链表的进一步探究

news2025/1/13 17:29:19

茕茕白兔,东走西顾,衣不如新,人不如故

往期回顾:

数据结构——双向链表 

数据结构——单链表

数据结构——顺序表

文章目录

如何判断一个链表是否为环形链表 

环形链表的判断的深入探究

例1:沸羊羊追美羊羊

例2:灰太狼追懒羊羊

判断方法结论总结

环形链表的入口结点

源码及思路

另辟蹊径


  大家好,我是纪宁。这篇文章将深入探究环形链表。 

  环形链表的形成是因为当单链表最后一个结点的指针域没有指向空,而指向了前面的任意一个结点。单链表链表一旦成环,在遍历的时候就会造成在一个区域里循环,那么对于环形链表的探究也应运而生。

  在现实情况中,链表不可能这么短,所以如何判断链表是否成环,链表在何处成环,就成为了我们学习链表之后必须要研究的问题,也成为了面试题和笔试题中最喜欢考察的部分。

  在上一篇链表刷题总结快慢指针的时候,博主曾粗略的谈到了环形链表的概念。这篇文章,博主带大家深入探究并总结,也算是博主自我内核的进一步提升吧。上篇文章的链接:快慢指针

如何判断一个链表是否为环形链表 

  首先,要明白环形链表的特征,如果定义一个指针去维护环形链表,它会在特定的位置进入环形链表内部,然后一直在环形链表内部循环遍历,这里如果使用快慢指针的话,在思路上是比较好理解的:当一个快指针和一个慢指针一起从头结点往后走,如果链表存在环,那么快指针首先进入环中,而慢指针则后入环,快指针在一直绕环循环的时候,一定能追上慢指针。如果没有追上的话,说明快指针一直向后走,直到 NULL,则说明这个链表不是环形指针。

  代码如下

bool hasCycle(struct ListNode *head) {
    struct ListNode *slow=head;
    struct ListNode *fast=head;
    while(fast&&fast->next)
    {
        slow=slow->next;
        fast=fast->next->next;
        if(fast==slow)
            return true;
    }
    return false;
}

  此代码将快指针定义为 fast ,慢指针定义为 slow,快指针每次向连链表末走两格,慢指针每次向链表末走一格(这里的走几格意思就是指针指向当前结点的后第几个结点),当两个指针相遇的时候,就说明这个链表带环。

环形链表的判断的深入探究

  上段文章提到,快指针每次走两格,慢指针每次走一格,这里我们不禁有一些思考:真的能恰好追上吗?不会正好错过了吗?当快指针每次走三格,慢指针每次走一格呢?分析下面这几个例子。

例1:沸羊羊追美羊羊

  沸羊羊跑的比较快,那么在无止尽的环中,它和美羊羊真的能正好相遇吗?

  假设当美羊羊进入环中的时候,早已进入环中的沸羊羊和美羊羊在圆周上的距离为 N,而沸羊羊一次走的距离为 2,美羊羊一次走的距离为 1,沸羊羊每次比美羊羊多走的距离为 1 。

走 1 次后,他们俩之间的距离为  N-1 

走 2 次后,他们俩之间的距离为  N-2

......

走 N-2 次后,他们俩之间的距离为 2

走 N-1 次后,他们俩之间的距离为 1

走 N 次后,他们俩之间的距离就为0,沸羊羊成功完美追上了美羊羊。

  即使追上了,美羊羊就会接受沸羊羊吗?(开个玩笑) 

  这个例子说明了,快指针每次走两格,慢指针每次每次走一格,确实能相遇,可以用来判断环形链表,那么如果快指针每次走三格呢?

例2:灰太狼追懒羊羊

  众所周知,红太狼爱吃羊,却从来没吃到过,但她有一个最疼爱她的老公灰太狼。灰太狼每天日复一日的想要抓到羊讨老婆换新,其中他最爱抓的羊,也是他觉得最好抓的羊,就是懒洋洋。

  灰太狼和懒洋洋同时出发,灰太狼每次走三格,懒洋洋每次走一格,他们能完美相遇吗?

  当懒洋洋进入环中时,灰太狼和它的举例为 M,环的长度为C,而懒洋洋每次走的距离为 1,灰太狼每次走的距离为 3,灰太狼比懒洋洋每次多走2灰太狼能否和懒洋洋完美相遇并抓到懒洋洋呢?

  这种情况就和上面沸羊羊追美羊羊不一样了,无论沸羊羊和美羊羊距离多远,每次距离缩小1,总有一天距离会变为0。但灰太狼每次和懒洋洋的距离缩小 2,是否还会出现一直完美错过的情况。

假设灰太狼和懒洋洋之间的距离 M 为偶数,那么之间每次的距离为

M-2

M-4

......

2

0  

这种情况是一定能追上的。

那么如果他们之间的距离 M 为奇数,那么他们之间每次的距离为 

M-2

M-2

......

3

1

-1 

  当距离为-1时,灰太狼跑到了懒洋洋前面,那么灰太狼就刚好错过了懒洋洋,要进行下一圈追击。并且追击的距离与环的周长C有关,因为这次要追击的距离为 C-1。

  当 C-1 为偶数,即周长C为偶数时,则恰好可以追上懒洋洋,如果C-1 依然为奇数,即周长C为偶数时,那这次也追不上懒洋洋了。因为这次追击的距离为奇数,所以下次追击的距离依然为 C-1 为奇数,这就成了一个死循环,这种情况下灰太狼就永远也抓不到懒洋洋!

判断方法结论总结

当快指针每次走步,慢指针每次走步的时候,这两个指针一定能相遇,这种方法可以判断出链表是否为环形链表

当快指针每次走步,慢指针每次走步的时候,这两个指针是否相遇得取决于进入环中时两指针的距离,当这个距离为偶数时,则一定能相遇;当这个距离为奇数时,还要看环的周长,如果环的周长为奇数时,则可以相遇,当环的长度为偶数时,则永远不能相遇。这种方法不一定能判断出链表是否为环形链表

当快指针每次走四步时,会有更复杂的情况......

环形链表的入口结点

  当我们判断出了一个链表是否为环形链表的时候,还会思考,那这个链表是在哪个结点开始进入环的呢?

  从上面那个例子中可以得出,快指针每次走两格,慢指针每次走一格,快指针一定能追上慢指针,可以判断出链表是否带环。

  继续说美羊羊和沸羊羊那个例子,从上面的例子可以得之,当美羊羊进入环中时,沸羊羊一定能在一圈之内追上美羊羊。

  设环的周长为4,环前面的链表长度为 L ,沸羊羊和美羊羊的相遇点与入环点的距离为 X,因为沸羊羊在美羊羊入环前可能已经绕环好几圈了,就设沸羊羊在和美羊羊相遇前已经在环里走了N 圈。

  沸羊羊的速度是美羊羊的 2 倍,所以他们走的距离也是 2 倍关系

美羊羊走的距离为

沸羊羊走的距离为

列出等式为

得出 L 的值为

化简得

  (N-1)*C 为沸羊羊在环里走的整圈的路程,而 (C-X) 为环的长度减去环的入口点到环的相遇点之间的长度。这个公式说明了一件事,当两个速度相同的指针,一个从起点开始走,一个从相遇点开始走,他们一定会在环入口点相遇(可能相遇前一个指针已经绕环很多圈了)。 

源码及思路

  首先,要找到快慢指针相遇的点,如果快慢指针没有相遇的话,则说明这个链表没有环。其次,重新定义两个速度相同的指针,一个从快慢指针的相遇点开始绕着环走,一个指针从起点开始走,这两个指针相遇的地方,就是环的入口结点。

struct ListNode *detectCycle(struct ListNode *head) {
    struct ListNode *fast=head,*slow=head;
    while(fast&&fast->next)
    {
        fast=fast->next->next;
        slow=slow->next;
        if(fast==slow)//相遇了
        {
            struct ListNode*cur1=head;
            struct ListNode*cur2=fast;
            while(cur1!=cur2)
            {
                cur1=cur1->next;
                cur2=cur2->next;
            }
            return cur1;
        }
    }
    return NULL;
}

  这是力扣上一道中等难度的题,但是只要分析思路正确了,写代码就是分分钟的问题。

另辟蹊径

  第一种思路,在很多人看来,特别是第一次接触这道题的人,比较难理解一点,那这里还有一种思路较为简单的方法,但简单的思路通常伴随着较为复杂的代码,可以说各有取舍吧,大家按需选择。

  在此之前,博主在这里再介绍一个经典题目:

判断两个链表是否为相交链表,并找出他们的第一个共同结点

  相交链表,即两个链表有重叠的公共结点,他们一旦有公共结点重叠,那么这个结点后面的结点就一定会重合。

  判断环形链表的思路是,两个链表各定义一个指针来维护,先遍历计算出链表的长度,顺便比较链表最后一个结点是否重合,如果不重合,则肯定不是相交结点。先让较长链表的指针向前走 差值 步,然后两个指针同时走,直到有一个相同的结点,这个结点就是相交链表的第一个共同结点。

  回到刚开始的问题,如何另辟蹊径找到环形链表的入口呢?既然刚才将了相交链表的思路,那肯定一想就直到:转化为相交链表求。

  如图,在找到相遇点后,将相遇点断开。红色部分为一个链表,绿色部分为一个链表,他们组成一个相交链表,相交链表的第一个结点就是环形链表的入环结点。 

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {
    int lenA=1,lenB=1;//求相交链表的第一个结点
    struct ListNode*curA=headA,*curB=headB;
    while(curA->next!=NULL)
    {
        curA=curA->next;
        lenA++;
    }
    while(curB->next!=NULL)
    {
        curB=curB->next;
        lenB++;
    }
    if(curA->val!=curB->val)
    {
        return NULL;
    }
    int dis=abs(lenA-lenB);//求差值绝对值
    struct ListNode*longnode=headA,*shortnode=headB;
    if(lenB>lenA)
        {
            longnode=headB;
            shortnode=headA;
        }
    while(dis--)
    {
        longnode=longnode->next;
    }
    while(longnode!=NULL)
    {
        if(longnode==shortnode)
        {
            return longnode;
        }
        longnode=longnode->next;
        shortnode=shortnode->next;
    }
    return NULL;
}
struct ListNode *detectCycle(struct ListNode *head) {
    struct ListNode *slow=head;
    struct ListNode *fast=head;
    while(fast&&fast->next)
    {
        slow=slow->next;
        fast=fast->next->next;
        if(fast==slow)//求相遇点
        {
            struct ListNode *cur=fast->next;
            fast->next=NULL;
            return getIntersectionNode(cur,head);
        }
             
    }
    return NULL;
}

  这种方法虽然思路简单,但代码比较长,代码变长,就容易出错,所以说每种方法都是各有利弊。 

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

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

相关文章

多线程案例(3)

文章目录 多线程案例三三、 定时器 大家好,我是晓星航。今天为大家带来的是 多线程案例三 相关的讲解!😀 多线程案例三 三、 定时器 定时器是什么 定时器也是软件开发中的一个重要组件. 类似于一个 “闹钟”. 达到一个设定的时间之后, 就…

AgileBoot - 全栈项目启动

AgileBoot-Back-End: 基于Ruoyi做了大量重构优化的基础快速开发框架。采用Springboot Vue 3 Mybatis Plus 更面向对象的业务建模 面向生产的项目。(非玩具项目) 首先克隆代码,同是克隆前端和后端的代码。 前端代码启动: np…

机器学习---概述(二)

文章目录 1.模型评估1.1 分类模型评估1.2 回归模型评估 2. 拟合2.1 欠拟合2.2 过拟合2.3 适当拟合总结: 3.深度学习3.1层次(Layers):3.2 神经元(Neurons):3.3 总结 1.模型评估 模型评估是机器学…

【2种方法,jmeter用一个正则提取器提取多个值!】

jmeter中,用json提取器,一次提取多个值,这个很多人都会。但是,用正则提取器一次提取多个,是否可以呢? 肯定,很多人都自信满满的说,可以!形如:token":&q…

MC0111配速MC0112白日梦Ⅰ

MC0111配速 难度: 白银 时间限制:1秒 占用内存:128M 小码哥参加了学校的定向越野比赛,赛完后,他踌躇满志地拿着自己的成绩单,看着一段段的数据,想算一下自己整场比赛的平均配速是多少。…

无涯教程-Lua - 嵌套if语句函数

在Lua编程中,您可以在另一个if or else if语句中使用一个if or else if语句。 nested if statements - 语法 嵌套if 语句的语法如下- if( boolean_expression 1) then--[ Executes when the boolean expression 1 is true --]if(boolean_expression 2)then--[ Ex…

【腾讯云 Cloud Studio 实战训练营】基于Cloud Studio 通过Java实现和公众号的快速对接

目录 一、Cloud Studio是什么 1.1 Cloud Studio介绍 1.2 Cloud Studio功能特点 1.3 Cloud Studio的好处 二、实战案例 2.1 创建开发环境 2.2选择开发模板 2.3 代码编写 2.3.1 引入依赖包 2.3.2 创建Models配置类 2.3.3 创建测试类demo.java 三、使用总结 今天通过J…

数论—换元法

0x00 前言 换元法指将一个式子看做一个整体,进行整体运算,从而达到简化的目的。 0x01 例题: 1.求所有整数n,使得n1|n25 2.求所有整数n,使得n-2|n5n 同样使用n-2去换元即可。 3.求所有的整数n,使用n-1|…

c语言基础知识帮助理解(函数递归详解)

"从前有座山,山里有座庙,庙里有个老和尚和一个小和尚。有一天老和尚对小和尚说:“从前有座山.山里有座庙,庙里有个老和尚和一个小和尚,有一天老和尚对小和尚说:“从前有座山.山里有座庙,庙里有个老和尚…

uniapp返回

// 监听返回事件onNavigationBarButtonTap() {uni.showModal({title: 提示,content: 确定要返回吗?,success: (res) > {if (res.confirm) {uni.navigateBack({delta: 2})}}})},

Palo Alto Networks® PA-220R 下一代防火墙 确保恶劣工况下的网络安全

一、主要安全功能 1、每时每刻在各端口对全部应用进行分类 • 将 App-ID 用于工业协议和应用,例如 Modbus、 DNP3、IEC 60870-5-104、Siemens S7、OSIsoft PI 等。 • 不论采用何种端口、SSL/SSH 加密或者其他规避技术,都会识别应用。 • 使用…

opencv-34 图像平滑处理-2D 卷积 cv2.filter2D()

2D卷积是一种图像处理和计算机视觉中常用的操作,用于在图像上应用滤波器或卷积核,从而对图像进行特征提取、平滑处理或边缘检测等操作。 在2D卷积中,图像和卷积核都是二维的矩阵或数组。卷积操作将卷积核在图像上滑动,对每个局部区…

智慧消防解决方案

智慧消防解决方案引入了物联网云平台、云计算中心等创新技术,实现对消防物联网数据的实时采集、传输和分析处理。通过智能感知设备与云平台的紧密结合,该解决方案利用先进的物联网技术、移动互联网和云计算等先进技术手段,实现火灾报警信号的…

健身用什么耳机好、健身大神都戴的运动耳机推荐

随着健身逐渐成为一种普遍的生活状态,越来越多的运动者选择将音乐作为他们锻炼时的精神伴侣。一款好的运动耳机,佩戴的舒适性再到音质、续航、防水。大家肯定问到哪有什么运动耳机好用的呢?今天就为大家推荐几款合适的运动耳机 1、NANK南卡R…

差分隐私 MP-SPDZ框架安装

ubuntu虚拟机安装MP-SPDZ框架 1.下载安装包到虚拟机内 https://github.com/data61/MP-SPDZ/releases 安装git 报错Waiting for cache lock: Could not get lock /var/lib/dpkg/lock-frontend. It is held by process 4402(unattended-upgr) 解决方案 #杀死进程 sudo k…

重磅!三思LED显示闪耀第31届世界大运会五座赛事场馆

成都第31届世界大学生夏季运动会 2023中国成都。 世界大学生夏季运动会(以下简称“大运会”) 我们来了! 三思LED不仅以颇具想象力的艺术手法 亮相这场高规格体育盛会 更以卓越品质为此盛大赛事保驾护航! 揭秘|成都大运会会…

2023年电赛---运动目标控制与自动追踪系统(E题)关于网友的问题回复

前言 &#xff08;1&#xff09;各位私信问问题之前&#xff0c;看看自己的问题是不是在这个里面再问&#xff01; &#xff08;2&#xff09; <1>2023年8月3日&#xff0c;10点25分。增加第三问的细节回答。 <2>2023年8月3日&#xff0c;10点45分。更新关于舵机安…

HTML中元素和标签有什么区别?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 标签&#xff08;Tag&#xff09;⭐元素&#xff08;Element&#xff09;⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启航前端之旅 欢迎来到前端入门之旅&a…

使用爬虫代理IP速度慢是什么原因?

你们有没有遇到过使用爬虫代理IP速度慢的问题呢&#xff1f;相信很多使用爬虫抓取的人都曾经陷入过这个烦恼&#xff0c;今天我们就来聊聊这个话题。 首先&#xff0c;我们得明白为什么爬虫代理IP速度会变得慢。其实&#xff0c;原因有很多&#xff0c;比如代理服务器过多的连接…

高中生台灯什么牌子好?挑选不好可能会伤眼!

台灯是现在家庭都在使用的照明灯具&#xff0c;对于晚上工作学习、看书休闲都有很大的裨益&#xff0c;尤其是正在上学阶段的孩子&#xff0c;更是必备可少的学习“伴侣”。但是在挑选台灯的过程中也要注重护眼效果&#xff0c;在此基础上才是讨论性价比的问题&#xff0c;如果…