【数据结构与算法 刷题系列】求带环链表的入环节点(图文详解)

简介: 【数据结构与算法 刷题系列】求带环链表的入环节点(图文详解)

一、问题描述

原题链接   142. 环形链表 II - 力扣(Le142. 环形链表 II - 力扣(Le

二、解题思路

方法一:数学公式推导法

预备知识

此方法的数学推导建立在判断链表是否带环的基础算法上,推荐阅读前置文章

点击下方文字

【数据结构与算法 刷题系列】判断链表是否有环-CSDN博客

如图,通过快慢指针法得到两个指针相遇位置时

假设:
  • 链表入环前的长度为L
  • 环的长度为C
  • 快慢指针相遇节点为meet
  • 环的入口与相遇节点meet的距离为N
  • 相遇时,快指针已经在环内走了X圈(X>=1,快指针至少比慢指针都走一圈才能追上)
推导过程:

在meet相遇点

慢指针移动距离为L+N

快指针移动距离为L+X*C+N

 

另外,快指针移动距离是快指针的两倍

快指针也可以写成2(L+N)

将两条公式结合起来

2(L+N)=L+X*C+N

化简

L+N=X*C

L=X*C-N

L=(X-1)*C+C-N    

最终得到的公式:

L=(X-1)*C+C-N    

该公式在图中说明的问题:

链表入环前的长度L

与相遇点到环入口的距离再加(X-1)圈      ——X最少为1,所以X-1至少为0

是相等的

结论:

如果两个指针分别从链表起始位置和相遇点meet开始移动,那么两个指针第一次相遇的节点就是环的入口

方法二:转换为相交链表问题求解

此方法的数学推到建立在判断链表是否带环的基础算法上,推荐阅读

【数据结构与算法 刷题系列】相交链表-CSDN博客

该方法是将带环链表问题转换为相交链表问题,将问题降级处理

首先,依然要 求得快慢指针相遇交点

然后将取得该节点下一个节点地址,令其成为一个单独链表的首节点,断开链表

之后,这个问题就可以转换为相交链表问题来解决

三、代码实现

方法一实现代码

struct ListNode {
    int val;
    struct ListNode* next;
    
};
typedef struct ListNode ListNode;
 
//方法一:数学推理法
struct ListNode* detectCycle1(struct ListNode* head)
{
    ListNode* slow, * fast;
    slow = fast = head;
    ListNode* meet = NULL;
    while (fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        if (slow == fast)//先求得快慢指针相遇节点
        {
            meet = slow;
            while (meet != head)//两指针同时移动,相遇即是入口
            {
                meet = meet->next;
                head = head->next;
            }
            return meet;
        }
    }
    return NULL;
}

方法二实现代码

//求相交链表的交点的函数
struct ListNode* getIntersectionNode(struct ListNode* headA, struct ListNode* headB)
{
    ListNode* pcurA = headA;
    ListNode* pcurB = headB;
    int countA = 0;
    int countB = 0;
    while (pcurA)//求出链表长度
    {
        pcurA = pcurA->next;
        countA++;
    }
    while (pcurB)
    {
        pcurB = pcurB->next;
        countB++;
    }
    int tmp = abs(countA - countB);//长度差值
    ListNode* slow, * fast;
    if (countA < countB)
    {
        slow = headA;
        fast = headB;
    }
    else
    {
        slow = headB;
        fast = headA;
    }
    while (tmp--)//长链表先走差值的步数
    {
        fast = fast->next;
    }
    while (fast && slow)//同步比较
    {
        if (fast == slow)
            return fast;
        fast = fast->next;
        slow = slow->next;
    }
    return NULL;
}
struct ListNode* detectCycle(struct ListNode* head)
{
    ListNode* slow, * fast;
    slow = fast = head;
    ListNode* meet = NULL;
    while (fast && fast->next)
    {
        slow = slow->next;
        fast = fast->next->next;
        if (slow == fast)//先求得快慢指针相遇节点
        {
            meet = slow;
            ListNode* newhead = meet->next;
            meet->next = NULL;//将环断开,变成两个相交的链表
 
            return getIntersectionNode(head, newhead);
        }
    }
    return NULL;
}

总结

两种方法可以自行选用
第一种方法属于推理复杂,代码简单

第二种方法属于推理简单,代码实现细节复杂

可根据实际情况选择合适的方法

相关文章
|
1月前
|
算法
LeetCode第24题两两交换链表中的节点
这篇文章介绍了LeetCode第24题"两两交换链表中的节点"的解题方法,通过使用虚拟节点和前驱节点技巧,实现了链表中相邻节点的交换。
LeetCode第24题两两交换链表中的节点
|
1月前
|
Kubernetes 算法 调度
在k8S中,Scheduler使用哪两种算法将Pod绑定到worker节点?
在k8S中,Scheduler使用哪两种算法将Pod绑定到worker节点?
|
1月前
【刷题记录】链表的回文结构
【刷题记录】链表的回文结构
|
1月前
|
算法
【算法】合并两个有序链表(easy)——递归算法
【算法】合并两个有序链表(easy)——递归算法
【算法】合并两个有序链表(easy)——递归算法
|
1月前
|
存储 Java 程序员
"揭秘HashMap底层实现:从数组到链表,再到红黑树,掌握高效数据结构的秘密武器!"
【8月更文挑战第21天】HashMap是Java中重要的数据结构,采用数组+链表/红黑树实现,确保高效查询与更新。构造方法初始化数组,默认容量16,负载因子0.75触发扩容。`put`操作通过计算`hashCode`定位元素,利用链表或红黑树处理冲突。`get`和`remove`操作类似地定位并返回或移除元素。JDK 1.8优化了链表转红黑树机制,提升性能。理解这些原理能帮助我们更高效地应用HashMap。
30 0
|
1月前
|
算法
【数据结构与算法】共享双向链表
【数据结构与算法】共享双向链表
11 0
|
1月前
|
算法
【数据结构与算法】双向链表
【数据结构与算法】双向链表
10 0
|
1月前
|
算法
【数据结构与算法】循环链表
【数据结构与算法】循环链表
12 0
|
1月前
|
存储 算法
【数据结构与算法】链表
【数据结构与算法】链表
16 0
|
1月前
|
机器学习/深度学习
【刷题记录】相交链表
【刷题记录】相交链表