力扣142. 环形链表 II

题目

给定一个链表的头节点head,返回链表开始入环的第一个节点。 如果链表无环,则返回null。

链接:142. 环形链表 II - 力扣(LeetCode)

题解

方法一:设置两个指针,一个指针指向链表头结点,一个指针指向判环时快慢指针相遇的位置,然后两个指针同时走,它们会在环的第一个结点处相遇。

判断链表是否有环:力扣141. 环形链表_啊哈leelee~的博客-CSDN博客

证明如下。

假设链表起始结点head到环起始结点head'的距离为L,环起始结点head'到快慢指针相遇结点meet的距离为X,环的长度为C,则meet到head'的距离为C-X。

在判环时,慢指针slow走的路径长度为L+X,快指针fast走的路径长度为L+nC+X(n>=1)。而fast所走路径长度是slow所走路径长度的2倍,即L+nC+X=2*(L+X) ,展开得L= nC-X,也就是L=(n-1)C+C-X。即一个指针从head开始走,一个指针从meet开始走,这两个指针一定会在环起始结点相遇。

注意,在slow进环之前,fast可能已经在环里走了n圈了;slow进环之后,fast一定会在一圈之内追上slow,因为slow和fast之间的距离最大是C,而slow和fast每移动一次,它们俩之间的距离就会减1,所以在slow移动一圈之前fast一定会追上。

代码如下:

复制代码
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 (slow == fast)
        {
            //寻找环起始结点
            struct ListNode* meet = slow;
            while (head != meet)
            {
                head = head->next;
                meet = meet->next;
            }
            return meet;
        }
    }
    return NULL;
}

方法二:设置一个指针meet指向快慢指针相遇的位置,设置一个指针meetNext指向meet的next,然后把meet的next置空,把问题转化为求解两个链表相交结点。如下图所示,问题转化为求解以head为头结点的链表以meetNext为头结点的链表相交结点

求解链表相交结点:力扣160. 相交链表_啊哈leelee~的博客-CSDN博客

代码如下:

复制代码
struct ListNode* getIntersectionNode(struct ListNode* headA, struct ListNode* headB)
{
    struct ListNode* tailA = headA;
    struct ListNode* tailB = headB;
    int lenA = 1;
    int lenB = 1;
    while (tailA->next)
    {
        tailA = tailA->next;
        lenA++;
    }
    while (tailB->next)
    {
        tailB = tailB->next;
        lenB++;
    }
    if (tailA != tailB)
        return NULL;

    struct ListNode* longList = headA;
    struct ListNode* shortList = headB;
    if (lenA < lenB)
    {
        longList = headB;
        shortList = headA;
    }
    int gap = abs(lenA - lenB);
    while (gap--)
    {
        longList = longList->next;
    }
    while (longList != shortList)
    {
        longList = longList->next;
        shortList = shortList->next;
    }
    return longList;
}

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 (slow == fast)
        {
            struct ListNode* meet = slow;
            struct ListNode* meetNext = meet->next;
            meet->next = NULL;
            head = getIntersectionNode(head, meetNext);
            return head;
        }
    }
    return NULL;
}
相关推荐
MediaTea6 分钟前
AI 术语通俗词典:ID3 算法
人工智能·算法
Morwit7 分钟前
【力扣hot100】 221. 最大正方形
前端·算法·leetcode
呃呃本21 分钟前
算法题(矩阵)
线性代数·算法·矩阵
2301_7890156222 分钟前
Linux基础指令(一)
linux·运维·服务器·c语言·开发语言·c++·linux指令
呃呃本31 分钟前
算法题(普通数组、矩阵)
线性代数·算法·矩阵
计算机安禾37 分钟前
【计算机网络】第11篇:链路状态路由协议——Dijkstra算法与OSPF的分区架构
计算机网络·算法·架构
珂朵莉MM37 分钟前
第七届全球校园人工智能算法精英大赛-算法巅峰赛产业命题赛第二赛季优化题--遗传算法
人工智能·算法
gihigo199838 分钟前
严格耦合波分析计算光栅衍射效率算法
算法
可编程芯片开发39 分钟前
基于双Qlearning强化学习的温差发电系统电压动态补偿算法matlab仿真
算法·matlab·双qlearning强化学习·电压动态补偿·温差发电系统
Java成神之路-44 分钟前
【LeetCode 刷题笔记】69.x 的平方根 | 二分查找经典刷题题解
算法·leetcode