【链表OJ 10】环形链表Ⅱ(求入环节点)

前言: 

💥🎈个人主页:​​​​​​Dream_Chaser~ 🎈💥

✨✨刷题专栏:http://t.csdn.cn/UlvTc

⛳⛳本篇内容:力扣上链表OJ题目

目录

leetcode142. 环形链表 II

 1.问题描述

 2.代码思路

3.问题分析


leetcode142. 环形链表 II

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

 1.问题描述

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

 题解接口:

struct ListNode *detectCycle(struct ListNode *head) {
   
}

 2.代码思路

前提条件:

是fast走的路程是slow的2倍。

解题思路:

        第一步,先定义一个快指针fast以及一个慢指针slow,这里跟环形链表1的快慢指针的操作一致,不作详细说明。之后找到可以证明链表带环的相遇点,并定义meet指针指向slow或此时的fast。

       第二步:接着让head指针从链表第一个节点开始移动,meet指针从相遇点开始移动,然后它们将会在链表带环的入口处相遇。(这是这道题思考的方向,但是如何去证明呢?)

 代码实现:

struct ListNode *detectCycle(struct ListNode *head) {
   struct ListNode* fast=head,*slow=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;//如果链表无环,则返回 null
}

代码执行:

3.问题分析

结论证明:

        一个指针从相遇点(meet)走,一个指针从链表头(head)开始走,他们会在入口点(返回值)相遇。为什么?以下是证明:

假设:

链表头--环入口点距离:L

环入口点--相遇点距离:X

环的长度:C

依据题意求出slow指针所走过的距离,明显是L+X

然后思考一个问题:有没有可能slow进环转了几圈才追上?

        答:不可能! 1圈之内,fast必然追上slow,因为他们之间距离每次缩小1,不会错过,slow走1圈,fast都走了2圈了,肯定追上了。

        所以说可以简单的求出fast指针在环上走过的距离:L+C +X  ,并且根据

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

        L+X = C

        第一种情况:L=C-X -->可以求出链表头到环入口点距离

        试想一下,当L的距离越长,环的大小越小,那么L=C-X依旧成立吗?

        由图可得, 可得到第二个结论:L=(n-1)*C+C-X   (n-1)*C表示fast在环内转了(n-1)

        总结:无论是第一种情况,还是第二种情况,meet与head均会在入环处相遇。

        本篇到此结束,感谢你的来访与支持,如有错误,十分欢迎指正。

  • 50
    点赞
  • 46
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 98
    评论
C每日一练中的环形链表Ⅱ问题是指LeetCode刷题专栏中的第142题。该问题的目标是找到给定链表中的环的入口节。解决这个问题的一种常用方法是使用快慢指针。 快慢指针解法中,快指针的步数是慢指针的两倍,即慢指针一次走一步,快指针一次走两步。两个指针从链表的起始位置开始运行,如果链表中存在环,则快指针最终会追上慢指针,二者会在环中相遇。如果链表不带环,则快指针会率先走到链表的末尾。这种思路可以通过在纸上画图来更好地理解。 在解决环形链表Ⅱ问题时,我们可以按照以下步骤进行操作: 1. 定义快慢指针,初始时都指向链表的头节。 2. 快指针一次走两步,慢指针一次走一步,直到两个指针相遇或快指针到达链表末尾。 3. 如果快指针到达链表末尾,则说明链表中没有环,直接返回null。 4. 如果快慢指针相遇,则说明链表中存在环。 5. 重新定义一个指针,从链表的头节开始,与慢指针同时移动,每次移动一步,直到两个指针再次相遇。 6. 当两个指针再次相遇时,它们所指向的节就是环的入口节。 通过以上步骤,我们可以找到给定链表中环的入口节。注意,在实际编码中,需要考虑边界条件和特殊情况的处理。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [【LeetCode】每日一题:链表部分经典题型](https://blog.csdn.net/qq_63320529/article/details/130697933)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 98
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Dream_Chaser~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值