一、题目描述:一个链表中包含环,请找出该链表的环的入口结点。
二、解析
参见下图
假设相遇于D点,则快指针应该这时刚好套慢指针一圈(2倍速的必然结果,可以数学证明)
则此时快指针走的路程为 ‘AB + BCDEB + BD’ (用BCDEB表示一圈)(字母序表示方向,AB 表示A -> B)
慢指针走的路程为 ‘AB + BD’
由于S(快指针) = 2 * S(慢指针) (因为2倍速)(S表示总路程)
==> AB + BCDEB + BD = 2 * (AB + BD) -----(1)
==> AB + BD = BCDEB -----(2)
上式表明此时慢指针走过的全部路径刚好一圈
我们的目标是获得B点这一入环点
又根据一圈的关系 , 有一圈剩余部分
DB = BCDEB - BD -----(3)
联立式(2)(3), 有
AB = DB -----(4)
上式表示D到B的距离和A到B的距离是相等的
另一种说法,在慢指针从相遇点D继续向前走DB个长度,一个新指针从A起始点用同样速度 开始走,两个指针将会在B点相遇,而B点也正是我们想要的相遇点
作者:yzzz0_0
链接:https://leetcode-cn.com/problems/two-sum/solution/tu-jie-guo-cheng-by-yzzz0_0/
来源:力扣(LeetCode)
三、代码实现
public ListNode detectCycle(ListNode head) {
if(head == null)
return head;
ListNode slow = head;
ListNode fast = head;
boolean hasCycle = false;
while(fast.next != null && fast.next.next != null){
slow = slow.next;
fast = fast.next.next;
if(slow == fast){
hasCycle = true;
break;
}
}
// 若有环,找到入环开始的节点
if(hasCycle){
ListNode start = head;
while (start != slow){
start = start.next;
slow = slow.next;
}
return start;
}else {
return null;
}
}
/**
* Definition for singly-linked list.
* class ListNode {
* int val;
* ListNode next;
* ListNode(int x) {
* val = x;
* next = null;
* }
* }
*/