LeetCode | 一探环形链表的奥秘【快慢双指针妙解BAT等大厂经典算法题】

简介: 附详细算法图解和证明解说,教你使用快慢双指针一破环形链表的奥秘

前言

本文总结了力扣141.环形链表|以及142.环形链表||这两道有关环形链表的求解方案,去求证链表是否带环已经如何找出入环口的结点。
有关环形链表,在BAT等大厂面试中均有出现,一般是属于==中等难度==的题,需掌握

一、题目描述

原题传送门

给你一个链表的头节点 head ,判断链表中是否有环。

如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos 不作为参数进行传递 。仅仅是为了标识链表的实际情况。

如果链表中存在环 ,则返回 true 。 否则,返回 false 。

示例 1:
在这里插入图片描述

输入:head = [3,2,0,-4], pos = 1
输出:true
解释:链表中有一个环,其尾部连接到第二个节点。

示例 2:

在这里插入图片描述

输入:head = [1,2], pos = 0
输出:true
解释:链表中有一个环,其尾部连接到第一个节点。

示例 3:
在这里插入图片描述

输入:head = [1], pos = -1
输出:false
解释:链表中没有环。

提示:

  • 链表中节点的数目范围是 [0, 104]
  • -105 <= Node.val <= 105
  • pos 为 -1 或者链表中的一个 有效索引 。

二、思路分析与罗列

好,看完了题目描述,接下去我们来分析一下如何去求解这道题目
  • 首先对于此题,首先你要考虑的一点是怎么去判断一个链表是否有环?
  • 在一开始看题目的时候你可能想了很多的办法,但是当写代码的时候,发现又不对。有点同学就和我说:这不是很简单,搞一个指针,做一个遍历,若是若是这个指针又回到原来的交点,那不就是带环吗
  • 那我只能说,这个同学没有读清楚题目,题目并没有告诉你这个环的入口在哪里,你怎么去判断这个遍历的指针走了一圈呢?所以这都是无稽之谈,我们应该通过画图来进行分析
在这里插入图片描述
  • 从上述图中可以看到,我使用来一个叫做快慢指针,这其实是解决环形链表这种问题的最好手段,具体的思路就是让快慢指针同时遍历这个链表,快指针走两步,慢指针走一步,然后在不断遍历的过程中,若是两个指针重合了,说明链表带环,具体的证明我放在后面讲解
  • 我们先来看一下快慢指针是如何遍历的

牢记规则:快指针走两步,慢指针走一步

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

  • 好,通过上面的算法图示,相信你已经明白了快慢指针最后究竟是如何相遇的,这个光凭空想还真的不好想出来,但是我们画个图来分析一下,就非常地明确了。我们在下一模块来写写代码
class Solution {
public:
    bool hasCycle(ListNode *head) {
        ListNode* fast, *slow;
        fast = slow = head;

        while(fast && fast->next)       //判断奇数和偶数个结点的情况
        {
            slow = slow->next;
            fast = fast->next->next;

            if(fast == slow)
                return true;
        }
        return false;
    }
};
  • 可以看出,代码并不复杂,就是通过一个循环去让这两个快慢指针去遍历这个链表,若是它们相遇,则【return true】,若是循环遍历结束还是没有遇见,则说明链表不带环

三、证明:

1、【为何快指针每次走两步,慢指针走一步一定能相遇?】

  • 相信在看完我上面的一些简略分析后有些小伙伴一定会疑惑为什么快指针每次走两步,慢指针每次走一步一定能相遇
  • 首先我在这里做一个假设,就是当快指针已经入环,而慢指针刚好入环时,他们之间的距离相差N
在这里插入图片描述
  • 然后此时缩小快慢指针的宏观移动距离,然后观察两个指针的移动的是否会改变他们之间的距离
  • 首先记录下它们的第一次移动①
在这里插入图片描述
  • 然后是第二次移动,继续计算它们之间的距离
在这里插入图片描述
  • 于是我们可以得出来下面这个结论,快指针【fast】和慢指针【slow】在不断前进的过程中它们之间的距离是会不断缩短的,当它们之间的距离 = 0时,其实也就意味着它们相交了
  • 其实快指针就是一个不断在追逐慢指针的一个过程
在这里插入图片描述
  • 所以就可以证明这个结论——》【快指针每次走两步,慢指针走一步一定能相遇】

2、【快指针一次走3步,走4步,...n步行吗?】

  • 接下去我们再来证明一个问题,刚才快指针一次是走两步,一定能追上,那现在当这个快指针一次走3步、走4步能不能追得上呢?我们一起来分析一下
  • 情况有很多,我这里就拿【fast】走3步,【slow】走1步来进行一个证明
在这里插入图片描述
  • 那根据我们上一个问题的证明,这依旧设【fast】在环中当【slow】刚进环时两者之间的距离为N,然后就可以得到两者在追击时它们之间的距离每次会缩短2,然后就可以去计算它们可不可能相遇
  • 因为它们之间的距离每次缩短的长度是一致的,所以就需要看这个N的大小,也就是在环中【fast】和【slow】之间的距离,若是N为偶数,那最后它们之间的距离一定会减少到0,也就意味着相遇;若是N为奇数,那最后它们之间的距离一定会减少到-1,这就意味着【fast】追是追上【slow】,但是呢却刚好错过了,到达了它的前一位
在这里插入图片描述
  • 我们来做一个模拟,此时当【fast】和【slow】快相遇时,它们继续行走
在这里插入图片描述
在这里插入图片描述
  • OK,可以看到,它们确实是错过了,那此时它们之间距离是多少呢?设整个环的周长为C。此时它们之间的距离就变成了【C-1】
  • 此时就需要在【C-1】的基础上再去考虑它们会不会相遇,那其实也是一样的道理,当【C-1】为偶数时,它们会相遇,当【C-1】为奇数时,它们之间的距离依旧会变回【C-1】,==此时真的就变成一个环了,两个指针在里面绕来绕去就是不会相交,【fast】永远都追不上【slow】==
在这里插入图片描述
  • 那这个问题的其他示例其实也是一样,比如说快指针每次走5步、走6步,慢指针走个2步、3步,其实都是一个道理,只不过要进行一个取余运算,最后的结果还是一样,只要他们之间的距离为奇数,那么就永远追不上

【最后我们可以得出结论:当两个指针的相对速度为1时,一定能相遇;当两个指针的相对速度> 1时,则需要视两个指针之间的距离而定】

四、进阶:如何求出环的入口结点

  • 此进阶为是环形链表|的后一道题:point_right:142.环形链表||,可先看看原题

Way1:头结点到入口结点的距离剖析求证

  • 好,看完了如何去证明一些环内快慢指针相遇的问题,接下去我们继续深入,从我画的图里可看到,从链表的头结点过来有一个环,而且我标出了一个结点叫做【环的入口结点】,也就是那位同学说的从这个结点开始遍历去判断这个链表是否有环
  • 那我们该如何去求解这个环呢?这需要一个数学分析和推理验算的思维,听我给你讲一讲::book:
  • 首先我们做一个假设,从链表头结点开始到环形入口结点的距离为L,从入口结点到快慢指针相遇的距离为N,则从相遇处再到入口结点的距离就为C-N
  • 此时我们需要根据这些长度变量去写出慢指针和快指针到相遇为止走过的距离
在这里插入图片描述
  • 慢指针走过的路程根据我们上面的推论很好计算,就是【L + N】,而对于快指针来说,很多同学就会有所异或了,因为它是在追慢指针的一个过程,但是不知道它在经过了多少距离,于是有的同学就直接认为快指针走过的距离为【L + C + N】,也就是快指针在环中刚好走了一圈碰到慢指针,然后根据快慢指针的两倍关系,就得出L = C - N,其实这还是有所考虑不周,可能是我画的这个环误导他了,下面我在换一个环来看看
在这里插入图片描述
  • 可以看到,这环很小,我们设快指针走一步为半个环
在这里插入图片描述
  • 可以看到此时快指针已经走了一圈,要开始走第二圈
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
  • 然后经过一段时间后,他们终究会相遇,但此时快指针【fast】已经在环里走了好几圈了,因此这就是我们要考虑到的情况,当这个环很大的时候,可能环很大的时候,【fast】走上个一圈就可以遇到【slow】,但是当这个环很小的时候,【fast】就需要等待【slow】,于是它会在这个小环里一直走一直走,直到他们在环的入口点相遇为止
  • 然后我们就可以精确地分析出快指针所走的路程,即为【L + k*C + N】,k是快指针走了几圈,C是周长。于是我们就可以根据快慢指针的两倍关系得出从头结点到环形入口结点的距离L
在这里插入图片描述
  • 那其实这个快指针不是走了k圈,而是走了【k - 1】圈,因此我们可以将k用【k - 1】带入可得L = (k - 1)C - N,为了和原本的等式相同,于是加上C,变为【L = (k - 1)C + (C - N)】
  • 此时我们就可以拿这个式子去分析了,当【k = 1】时,也就是快指针走了一圈时,L就等于【C - N】,那也就是我们一开始算的从快慢指针相遇处到环形入口结点之间的距离,当【k > 1】时,就需要另加考虑,让快指针先走上k圈,然后再用环的周长 - N,此时才是L的长度

  • 那这个时候有同学问了,求出这段L的长度有什么用呢?对于,有什么用。其实我就是在证明在快慢指针已经相遇后要如何行走才可以到达这个环的入口现在我们得出一个表达式为【L = (k - 1)*C + (C - N)
在这里插入图片描述
  • 那我们可以在快慢指针的相遇处定义一个指针【cur1】,在结点处再定义一个指针【cur2】,然后让他们一直走一直走,通过这个环的大小来看出【cur1】会在这个环里转多少圈。我们来看一下代码
ListNode *detectCycle(ListNode *head) {
    ListNode* slow, *fast;
    fast = slow = head;
    while(fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        if(slow == fast)
        {
            ListNode* cur1 = fast;
            ListNode* cur2 = head;
            while(cur1 != cur2)
            {
                cur1 = cur1->next;
                cur2 = cur2->next;
            }
            return cur1;
        }
    }
    return NULL;
}
  • 可以看出,代码并不难写,只是我们在分析证明的时候花了很大的心思

Way2:环形链表转相交链表【秒不可言】

  • 有些同学可能一开始拿到这道题的时候想不到用这个数学推理的方法去求证环的入口点,我这里再给出一种方案,虽然比较抽象,但确实是【秒】啊!
在这里插入图片描述
  • 第一种方法是当这个快慢指针相遇之后,是采取了又定义两个指针,一个从相遇处出发,一个从头结点出发,然后直至它们相遇处便是环的入口结点。
  • 但是我们现在换个思路,就是在相遇处定义一个指针,然后将其【next】设置为一个新的链表头,然后再让相遇处的这个地方的【next】置为NULL,那么这个相遇处就相当于是尾结点,置为NULL就相当于是尾结点指向NULL
  • 此时我们一样的思路,也不需要考虑头结点要不要保存,其实这就变成了一道相交链表的题目,这是另一道LeetCode习题--》相交链表
在这里插入图片描述
  • 看了另外一题后,你就会明白这种方法有多少巧妙,不得不说,【杭神牛逼!!!】
  • 给出核心代码给你看看,整体的在下面放出。从运行来看,一样是可以过的
在这里插入图片描述
if(slow == fast)
{
    ListNode* meet = slow;
    ListNode* otherHead = meet->next;      //新的链表头
    meet->next = NULL;      //meet相当于尾结点,也就是尾结点指向空
    ListNode* meetNode = getIntersectionNode(head,otherHead);
    return meetNode;
}

五、疑难解惑:为什么快指针会在慢指针进入环内的第一圈就相遇?

  • 有些小伙伴可能还是对快指针为何会在慢指针进到环里但是还没有碰到环的出口时就会相遇,我们来继续探究一下:mag:
因为快指针一定是先进入环内的,然后慢指针才进到环内,然后当慢指针进入下一个入口时,==快指针走的一定是慢指针的两倍==,所以慢指针在没有进入到下一个入口处时,快指针在中间的某个位置一定和其相遇了

证明如下:

在快指针fast进入环口3时,它已经走了 k + n个结点,从图中可以清晰地看出,k为快指针和慢指针之间的距离,n为一个环的距离,而慢指针在进入环内相应地走了(k + n)/2个结点,从图中可以看出k是小于n的,所以(k + n)/2也一样是小于n的,即慢指针在进入环内一圈不到的距离就会和快指针相遇

所以慢指针走动的距离为L + N就够了,其不会再走第二圈

原理图

请添加图片描述

六、整体代码展示

1、环形链表|

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    bool hasCycle(ListNode *head) {
        ListNode* fast, *slow;
        fast = slow = head;

        while(fast && fast->next)       //判断奇数和偶数个结点的情况
        {
            slow = slow->next;
            fast = fast->next->next;

            if(fast == slow)
                return true;
        }
        return false;
    }
};

2、环形链表||

Way1

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    ListNode *detectCycle(ListNode *head) {
        ListNode* slow, *fast;
        fast = slow = head;
        while(fast && fast->next)
        {
            slow = slow->next;
            fast = fast->next->next;
            if(slow == fast)
            {
                ListNode* cur1 = fast;
                ListNode* cur2 = head;
                while(cur1 != cur2)
                {
                    cur1 = cur1->next;
                    cur2 = cur2->next;
                }
                return cur1;
            }
        }
        return NULL;
    }
};

Way2

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
private:
    ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
        ListNode* curA = headA, *curB = headB;      //遍历
        int lenA = 0, lenB = 0;     //记录两个链表的长度
        while(curA->next){
            lenA++;
            curA = curA->next;
        }

        while(curB->next){
            lenB++;
            curB = curB->next;
        }

        if(curA != curB)
            return NULL;        //若两个结点的地址不同,则表示不相交,return NULL

        ListNode* longer = headA, *shorter = headB;      //先假设链表A长于链表B
        if(lenB > lenA){            //若是假设错误则交换
            longer = headB;
            shorter = headA;
        }
            
        int gap = abs(lenA - lenB);        //求出两个链表的长度差
        while(gap--)
            longer = longer->next;      //先让长的链表走gap步,使得两链表位于同一起跑线

        while(longer != shorter)        //一同往后走,寻找两链表的交点(此时一定相交) 
        {
            longer = longer->next;
            shorter = shorter->next;
        }
        return longer;
    }
public:
    ListNode *detectCycle(ListNode *head) {
        ListNode* slow, *fast;
        fast = slow = head;
        while(fast && fast->next)
        {
            slow = slow->next;
            fast = fast->next->next;
            if(slow == fast)
            {
                ListNode* meet = slow;
                ListNode* otherHead = meet->next;      //新的链表头
                meet->next = NULL;      //meet相当于尾结点,也就是尾结点指向空
                ListNode* meetNode = getIntersectionNode(head,otherHead);
                return meetNode;
            }
        }
        return NULL;
    }
};

七、总结与提炼

  • 最后我们来总结一下本文所介绍的内容,本文我们探究了链表章节比较复杂的一种题型——==环形链表==,这也是BAT等大厂在面试时很喜欢出的算法题,不仅仅是会让你手写代码,而且还会让你现场做个证明,因此对于上述的一些证明,希望大家也可以搞懂
  • 我们知道,编程的核心是算法,算法的本质是数学,你数学好了,逻辑思维就能强,面对一些棘手的推理算法题时才能游刃有余
以上就是本文所要描述的所有内容,感谢您对本文的观看,如有疑问请于评论区留言或者私信我都可以:four_leaf_clover:
相关文章
|
3月前
【力扣】-- 移除链表元素
【力扣】-- 移除链表元素
42 1
|
3月前
Leetcode第21题(合并两个有序链表)
这篇文章介绍了如何使用非递归和递归方法解决LeetCode第21题,即合并两个有序链表的问题。
56 0
Leetcode第21题(合并两个有序链表)
|
2月前
|
存储 算法 Java
leetcode算法题-有效的括号(简单)
【11月更文挑战第5天】本文介绍了 LeetCode 上“有效的括号”这道题的解法。题目要求判断一个只包含括号字符的字符串是否有效。有效字符串需满足左括号必须用相同类型的右括号闭合,并且左括号必须以正确的顺序闭合。解题思路是使用栈数据结构,遍历字符串时将左括号压入栈中,遇到右括号时检查栈顶元素是否匹配。最后根据栈是否为空来判断字符串中的括号是否有效。示例代码包括 Python 和 Java 版本。
|
3月前
LeetCode第二十四题(两两交换链表中的节点)
这篇文章介绍了LeetCode第24题的解法,即如何通过使用三个指针(preNode, curNode, curNextNode)来两两交换链表中的节点,并提供了详细的代码实现。
32 0
LeetCode第二十四题(两两交换链表中的节点)
|
3月前
|
索引
力扣(LeetCode)数据结构练习题(3)------链表
力扣(LeetCode)数据结构练习题(3)------链表
107 0
|
4月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
5月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
65 6
|
5月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
133 2
|
2月前
|
机器学习/深度学习 人工智能 自然语言处理
280页PDF,全方位评估OpenAI o1,Leetcode刷题准确率竟这么高
【10月更文挑战第24天】近年来,OpenAI的o1模型在大型语言模型(LLMs)中脱颖而出,展现出卓越的推理能力和知识整合能力。基于Transformer架构,o1模型采用了链式思维和强化学习等先进技术,显著提升了其在编程竞赛、医学影像报告生成、数学问题解决、自然语言推理和芯片设计等领域的表现。本文将全面评估o1模型的性能及其对AI研究和应用的潜在影响。
57 1
|
4月前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口