链表 - 判断链表是否有环以及环的入口

给一个链表,若其中包含环,请找出该链表的环的入口结点,否则,输出null。

一. 什么是链表的环?

单链表出现循环的情况就是链表的环。
所以,寻找环的入口就是找到循环开始的地方。

二. 解决方案

  • 快慢指针
    • 理解该方法需要我们推导一条原理
    • 如图所示,x为链表入口到环入口的距离,y为环入口到快慢指针相遇点的距离,z为相遇点到环入口的距离。
    • 同时,环的总长度为L,即L = y + z
    • 设置快指针,每次走2个节点,记为2S。
    • 设置慢指针,每次走1个节点,记为1S。


    • 化简得到:x = (n - 1)L + z
    • 其中n是快指针比慢指针多走的循环数,当n = 1时,x = z,也就是说,当快指针只比慢指针多走一圈就相遇的话,链表入口到环入口的距离=相遇点到环入口的距离。当n != 1 时,道理一样,只不过快指针跑多几圈而已。
    • 正是基于上面的结论,可以在快慢指针第一次相遇的地方重置快指针的位置,使快指针回到链表入口,慢指针不动。然后,两个指针以相同的速度在运动,再次相遇的地方就是环入口。
function EntryNodeOfLoop(pHead)
{
    let fast = pHead;
    let slow = pHead;
    while(fast && fast.next){
        fast = fast.next.next;// 快指针一次走两步
        slow = slow.next;// 慢指针一次走一步
        if(fast == slow){ // 第一次相遇重置快指针的位置以及速度
            fast = pHead;
            while(fast != slow){
                fast = fast.next;
                slow = slow.next;
            }
            return fast;// 再次相遇的地方就是环入口
        }
    }
    return null;
}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 题意:给定一个单向链表,求判断该链表是否为带环链表并求出该环的入口点 来源地址:Chasiny 例如下图,一个带环...
    Chasiny阅读 821评论 0 2
  • 题目描述 在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针。 例如,...
    Mereder阅读 5,243评论 0 2
  • 给一个链表,若其中包含环,请找出该链表的环的入口结点,否则,输出null。 分析:如何判断有环?如何找到环的入口节...
    ShawnCaffeine阅读 3,677评论 2 2
  • 搞懂单链表常见面试题 Hello 继上次的 搞懂基本排序算法,这个一星期,我总结了,我所学习和思考的单链表基础知识...
    醒着的码者阅读 4,618评论 1 45
  • 题目:如何判断一个单链表是否有环?若有环,如何找出环的入口节点。 一、单链表是否有环 思路分析: 单链表有环,是指...
    sherrysack阅读 712评论 0 0