LeetCode 142题解|环形链表II的快慢指针法(含数学证明)

题目如下:

解题过程如下:

思路:快慢指针在环里一定会相遇,相遇结点到入环起始结点的距离 == 链表头结点到入环起始结点的距离(距离看从左往右的方向,也就是单链表的方向),从链表头结点和相遇结点遍历,只要结点一样,那么这个结点就是入环起始结点。

示例1、示例2为例,

示例1:相遇结点到入环起始结点的距离1 == 链表头结点到入环起始结点的距离1

示例2:相遇结点到入环起始结点的距离0 == 链表头结点到入环起始结点的距离0

完整代码如下:

c 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
typedef struct ListNode ListNode;
struct ListNode *detectCycle(struct ListNode *head) {
    //快慢指针
    ListNode* slow = head;
    ListNode* fast = head;
    while (fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        //快慢指针相遇
        if (slow == fast)
        {
            //链表头结点和相遇结点开始往后遍历,结点一样,这个结点就是入环起始结点
            ListNode* pcur = head;
            while (slow != pcur)
            {
                slow = slow->next;
                pcur = pcur->next;
            }
            return slow;
        }
    }
    //fast == NULL 或 fast->next == NULL,跳出循环,说明没有环
    return NULL;
}

试着证明:

为什么在带环链表中,链表的头结点和快慢指针相遇结点到入环起始结点的距离相等?

假设:链表的头结点到入环起始结点的距离是L,环的周长是R,若slow刚刚入环时fast已经在环里绕了n圈了(n至少为1,因为fast先进环中到M点,后又和slow在M点相遇),入环起始结点到相遇结点之间的距离是X。

慢指针进环后,快指针肯定会在慢指针走一圈之内追上慢指针。因为在快慢指针都进环之后,快慢指针之间的距离最多就是一个环的周长,快指针每追击1次,二者之间的距离就会缩小1步,所以,在慢指针移动一圈之前,快指针一定会追上慢指针。

若已经相遇,快慢指针走过的路程:

慢指针 = L + X

快指针 = L + X + nR

由于快慢指针走过的路程之间的关系2 * 慢指针 = 快指针,得出L = nR - X = (n - 1)R + R - X,式子L = (n - 1)R + R - X(n为1,2,3,4,......,n的大小取决于环的大小,环越大n越小)中,(n - 1)R表示绕(n - 1)圈,取极端情况,n = 1时,式子最终可以看成L = R - X,即slow指针从链表起始位置开始向后遍历,fast指针在相遇点开始环绕,最终一定会在入环起始结点相遇;也就是说,在带环链表中,链表的头结点和快慢指针相遇结点到入环起始结点的距离相等。

相关推荐
你撅嘴真丑23 分钟前
字符环 与 变换的矩阵
算法
想放学的刺客24 分钟前
单片机嵌入式试题(第29期)嵌入式系统的电源完整性设计与去耦电容选型。抗干扰设计与EMC合规性
c语言·stm32·嵌入式硬件·物联网·51单片机
早点睡觉好了39 分钟前
重排序 (Re-ranking) 算法详解
算法·ai·rag
gihigo199843 分钟前
基于全局自适应动态规划(GADP)的MATLAB实现方案
算法
酒鼎1 小时前
学习笔记(4)HTML5新特性(第3章)- WebSocket
笔记·学习·html5
念越1 小时前
数据结构:栈堆
java·开发语言·数据结构
dear_bi_MyOnly1 小时前
【多线程——线程状态与安全】
java·开发语言·数据结构·后端·中间件·java-ee·intellij-idea
-Springer-2 小时前
STM32 学习 —— 个人学习笔记2-2(新建工程)
笔记·stm32·学习
ctyshr2 小时前
C++编译期数学计算
开发语言·c++·算法
tb_first2 小时前
万字超详细苍穹外卖学习笔记4
java·spring boot·笔记·学习·spring·mybatis