问题

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

为了表示给定链表中的环,我们使用整数 leetcode-142:环形链表Ⅱ - 图1 来表示链表尾连接到链表中的位置(索引从 0 开始)。 如果 leetcode-142:环形链表Ⅱ - 图2 是 -1,则在该链表中没有环。注意,pos 仅仅是用于标识环的情况,并不会作为参数传递到函数中

不允许修改给定的链表

进阶:你是否可以使用 leetcode-142:环形链表Ⅱ - 图3 空间解决此题?

示例1:
leetcode-142:环形链表Ⅱ - 图4
输入:head = [3,2,0,-4], pos = 1
输出:返回索引为 1 的链表节点
解释:链表中有一个环,其尾部连接到第二个节点

示例2:
leetcode-142:环形链表Ⅱ - 图5
输入:head = [1,2], pos = 0
输出:返回索引为 0 的链表节点
解释:链表中有一个环,其尾部连接到第一个节点

思路

此处需要注意两点:

  • 判断链表是否环
  • 如果有环,如何找到这个环的入口

判断链表是否有环

快慢指针法:分别定义leetcode-142:环形链表Ⅱ - 图6leetcode-142:环形链表Ⅱ - 图7指针,从头结点出发,leetcode-142:环形链表Ⅱ - 图8指针每次移动两个节点,leetcode-142:环形链表Ⅱ - 图9指针每次移动一个节点,如果leetcode-142:环形链表Ⅱ - 图10leetcode-142:环形链表Ⅱ - 图11指针在途中相遇 ,说明这个链表有环

为什么leetcode-142:环形链表Ⅱ - 图12走两个节点,leetcode-142:环形链表Ⅱ - 图13走一个节点,有环的话,一定会在环内相遇呢,而不是永远的错开呢?

  • leetcode-142:环形链表Ⅱ - 图14指针一定先进入环中,如果leetcode-142:环形链表Ⅱ - 图15指针和leetcode-142:环形链表Ⅱ - 图16指针相遇的话,一定是在环中相遇,这是毋庸置疑的

如果有环,如何找到这个环的入口

假设从头结点到环形入口节点 的节点数为leetcode-142:环形链表Ⅱ - 图17,环形入口节点到leetcode-142:环形链表Ⅱ - 图18指针与leetcode-142:环形链表Ⅱ - 图19指针相遇节点节点数为leetcode-142:环形链表Ⅱ - 图20,从相遇节点再到环形入口节点节点数为leetcode-142:环形链表Ⅱ - 图21,如图所示:
image.png
那么相遇时:leetcode-142:环形链表Ⅱ - 图23指针走过的节点数为:leetcode-142:环形链表Ⅱ - 图24leetcode-142:环形链表Ⅱ - 图25指针走过的节点数:leetcode-142:环形链表Ⅱ - 图26leetcode-142:环形链表Ⅱ - 图27leetcode-142:环形链表Ⅱ - 图28指针在环内走了leetcode-142:环形链表Ⅱ - 图29圈才遇到leetcode-142:环形链表Ⅱ - 图30指针,leetcode-142:环形链表Ⅱ - 图31为一圈内节点的个数。因为leetcode-142:环形链表Ⅱ - 图32指针是一步走两个节点,leetcode-142:环形链表Ⅱ - 图33指针一步走一个节点, 所以leetcode-142:环形链表Ⅱ - 图34指针走过的节点数 = leetcode-142:环形链表Ⅱ - 图35指针走过的节点数 * 2:leetcode-142:环形链表Ⅱ - 图36,即leetcode-142:环形链表Ⅱ - 图37

因为要找环形的入口,那么要求的是leetcode-142:环形链表Ⅱ - 图38,因为leetcode-142:环形链表Ⅱ - 图39表示头结点到环形入口节点的的距离。所以要求leetcode-142:环形链表Ⅱ - 图40,将leetcode-142:环形链表Ⅱ - 图41单独放在左面:leetcode-142:环形链表Ⅱ - 图42,再从leetcode-142:环形链表Ⅱ - 图43中提出一个leetcode-142:环形链表Ⅱ - 图44来,整理之后如下:leetcode-142:环形链表Ⅱ - 图45, 注意这里n一定是大于等于1的,因为leetcode-142:环形链表Ⅱ - 图46指针至少要多走一圈才能相遇leetcode-142:环形链表Ⅱ - 图47指针

这个公式说明什么呢?

  • leetcode-142:环形链表Ⅱ - 图48时,意味着leetcode-142:环形链表Ⅱ - 图49指针在环形里转了一圈之后,就遇到了leetcode-142:环形链表Ⅱ - 图50指针了。当leetcode-142:环形链表Ⅱ - 图51时,公式就化解为leetcode-142:环形链表Ⅱ - 图52,这就意味着,从头结点出发一个指针,从相遇节点也出发一个指针,这两个指针每次只走一个节点, 那么当这两个指针相遇的时候就是 环形入口的节点,也就是在相遇节点处,定义一个指针leetcode-142:环形链表Ⅱ - 图53,在头结点处定一个指针leetcode-142:环形链表Ⅱ - 图54,让leetcode-142:环形链表Ⅱ - 图55leetcode-142:环形链表Ⅱ - 图56同时移动,每次移动一个节点, 那么他们相遇的地方就是环形入口的节点
  • leetcode-142:环形链表Ⅱ - 图57时,leetcode-142:环形链表Ⅱ - 图58指针在环形转leetcode-142:环形链表Ⅱ - 图59圈之后才遇到leetcode-142:环形链表Ⅱ - 图60指针,其实这种情况和leetcode-142:环形链表Ⅱ - 图61的时效果是一样的,一样可以通过这个方法找到环形的入口节点,只不过,leetcode-142:环形链表Ⅱ - 图62指针在环里多转了leetcode-142:环形链表Ⅱ - 图63圈,然后再遇到leetcode-142:环形链表Ⅱ - 图64,相遇点依然是环形的入口节点

超时了,为什么啊???

  1. package leetcode;
  2. public class leetcode_142 {
  3. public ListNode detectCycle(ListNode head){
  4. ListNode fast = head, slow = head;
  5. while(fast != null && fast.next != null){
  6. slow = slow.next;
  7. fast = fast.next.next;
  8. }
  9. if (fast == slow){
  10. ListNode index1 = fast;
  11. ListNode index2 = head;
  12. while(index1 != index2){
  13. index1 = index1.next;
  14. index2 = index2.next;
  15. }
  16. return index2;
  17. }
  18. return null;
  19. }
  20. }

官方题解

  1. public class Solution {
  2. public ListNode detectCycle(ListNode head) {
  3. if (head == null) {
  4. return null;
  5. }
  6. ListNode slow = head, fast = head;
  7. while (fast != null) {
  8. slow = slow.next;
  9. if (fast.next != null) {
  10. fast = fast.next.next;
  11. } else {
  12. return null;
  13. }
  14. if (fast == slow) {
  15. ListNode ptr = head;
  16. while (ptr != slow) {
  17. ptr = ptr.next;
  18. slow = slow.next;
  19. }
  20. return ptr;
  21. }
  22. }
  23. return null;
  24. }
  25. }