首页 > 代码库 > 【LeetCode】Linked List Cycle II

【LeetCode】Linked List Cycle II

Given a linked list, return the node where the cycle begins. If there is no cycle, return null.

Follow up:
Can you solve it without using extra space?

 

首先不考虑Follow up的要求,那就比较简单了。

创建一个map,存放已经出现过的指针,直到next为空或者next指向的指针出现过。

class Solution {
public:
    map<ListNode *, bool> m;
    ListNode *detectCycle(ListNode *head) 
    {
        ListNode* p = head;

        while(p != NULL)
        {
            if(m.find(p) == m.end())
            {
                m.insert(map<ListNode *, bool>::value_type(p, true));
                p = p->next;
            }
            else
                return p;
        }
        return NULL;
    }
};

 

考虑不创建额外存储空间的情况。

有个不靠谱想法是,寻找一个与链表中的val都不同的val_unique,将所有遍历过的节点都置为val_unique,

直到next为空或者next指向的节点为val_unique。

虽然在平台上使用INT_MIN可以AC,但是INT_MIN是否就是unique,这是无法判断的。

上不靠谱代码:

class Solution {
public:
    ListNode *detectCycle(ListNode *head) 
    {
        ListNode* p = head;

        while(p != NULL)
        {
            if(p->val == INT_MIN)
                return p;
            else
            {
                p->val = INT_MIN;
                p = p->next;
            }
        }
        return NULL;
    }
};

 

最后参考了Discussion中提到的龟兔赛跑算法,经过理解实现,将思路与代码整理如下:

兔(fast)每次前进两步(若两步中出现NULL,则返回NULL).

龟(slow)每次前进一步(若出现NULL,则返回NULL).

当出现环路,则总有某时刻,兔会赶上龟(相遇),证明如下:

1、若兔子套圈之前在龟后方两步,则兔2龟1之后,兔子在龟后方一步,进入(2)

2、若兔子套圈之前在龟后方一步,则兔2龟1之后,兔子追上龟(相遇)

其他情况均可化归到以上两步。

假设从head到环入口entry步数为x,环路长度为y,相遇时离entry的距离为m

则兔:x+ay+m,龟:x+by+m  (a > b)

x+ay+m = 2(x+by+m)

整理得x+m = (a-b)y

以上式子的含义是,相遇时的位置(m)再前进x步,正好是y的整倍数即整圈。

现在的问题是怎样计数x。

利用head到entry的长度为x,只要龟从head每次前进一步,兔从相遇位置每次前进一步,

再次相遇的时候即环的入口。

class Solution {
public:

    ListNode *detectCycle(ListNode *head) 
    {
        if(head == NULL)
            return NULL;

        ListNode* fast = head;
        ListNode* slow = head;

        do
        {
            if(fast->next != NULL && slow->next != NULL)
            {
                fast = fast->next;
                if(fast->next != NULL)
                    fast = fast->next;
                else
                    return NULL;
                slow = slow->next;
            }
            else
                return NULL;
        }while(fast != slow);

        slow = head;
        while(fast != slow)
        {
            fast = fast->next;
            slow = slow->next;
        }
        return fast;
    }
};

这里的循环有个小技巧,用do...while而不是while,来保证第一次判断结束条件时兔子已经比龟多前进了一步。