题目描述
给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。
为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。 如果 pos 是 -1,则在该链表中没有环。注意,pos 仅仅是用于标识环的情况,并不会作为参数传递到函数中。
说明:不允许修改给定的链表。
方法:
快慢指针同时从head出发,设快指针走了n圈,因为快指针始终每次都比慢指针多走一步,则相遇时,快指针的路程:a+n(b+c)+b 慢指针的路程:a+b
a+n(b+c)+b = 2(a+b)
a = (n-1)(b+c) + c 即 从相遇点到入环点的距离b加上 n-1n−1 圈的环长(b+c),恰好等于从链表头部到入环点的距离a。
所以相遇时,再将fast指向head,和slow 每次向后移动一个位置。最终,它们会在入环点相遇。
/**
* Definition for singly-linked list.
* class ListNode {
* int val;
* ListNode next;
* ListNode(int x) {
* val = x;
* next = null;
* }
* }
*/
public class Solution {
public ListNode detectCycle(ListNode head) {
if(head == null || head.next == null){
return null;
}
ListNode fast = head;
ListNode slow = head;
while(true){
if(fast == null || fast.next == null){
return null;
}
fast = fast.next.next;
slow = slow.next;
if(slow == fast){
break;
}
}
fast = head;
while(slow != fast){
slow = slow.next;
fast = fast.next;
}
return fast;
}
}