龟兔算法---对环内相遇问题的理解

一、问题背景

141. 环形链表 - 力扣(LeetCode)

给定一个链表,如何去判断是否该链表中有环,示意图如下。

 图1:环形链表

图2:普通单链表

        解决该问题比较常用的方法就是快慢指针法,也就是所谓的龟兔赛跑算法的简化版:基本原理就是定义两个链表指针分别指向链表的头结点和第二个结点(注:在力扣中一般链表的头结点就是首元结点),在两个指针都进入链表环以后,总会相遇在同一个结点,反之如果是普通链表则指针最终会指向空。参考程序C++如下:

/**
 * struct ListNode { //链表结点的定义
 *     int val;
 *     ListNode *next;
 *     ListNode(int x) : val(x), next(NULL) {}
 * };
 */
class Solution {
public:
    bool hasCycle(ListNode *head) {
        if(head == NULL){ //判断边界条件
            return false;
        }
        ListNode* slow = head;
        ListNode* fast = head->next;
        while(fast != NULL && fast->next != NULL){ //当快指正不为空时执行循环
            if(slow == fast){       //快指正与慢指针套圈时有环
                return true;
            }
            slow = slow->next;
            fast = fast->next->next;
        }
        return false;
    }
};

        相信有部分小伙伴会对这一结论感到奇怪:为什么两个指针速度不一样,在进入环后一定就会相遇在同一结点呢?下面着重对这一个问题谈谈我个人的理解。

二、龟兔环内相遇的必然性

1、对问题的数学描述:

        若一个链表环有 n 个结点,存在两个指针:一个是慢指针,每步移动一个结点,记为 slow ,一个是快指针,每步移动两个结点,记为 fast ,为了简化问题,假设这两个指针从同一个起点出发,且该起点的标号为1。试证明两指针必然会相遇在同一结点,并求出此时走过的步数 i 应该满足的关系。

2、对问题的分析:

        该问题为证明性问题,我们可以分别求出经 i 步后 slow 与 fast 指针所指向结点的标号I_{s}I_{f},然后对标号相等的条件进行求解,若结果满足题意限制即上述结论得证。示意图绘制如下:

图3:经过 i 步后各个指针的示意图

 3、对问题求解

        针对 slow 指针,其速度为一步一格,初始值为1,标号建立如式3-1:

                                                        I_{s}=(i+1)\% n                                                   (3-1)

        针对fast指针,其速度为一步两格,初始值为1,标号建立如式3-2:

I_{f}=(2i+1)\% n                                                  (3-2)

        其中,\%表示取模运算,可能有的小伙伴凭着直觉会认为要对结点个数 n 分奇偶讨论,因为fast指针每次移动两格,但事实上,奇偶讨论的结果和式(3-2)一样,有兴趣的可以自己去验证。

        若要使得两个指针相遇,即达到同一结点,满足其对应标号相等:

I_{s}=I_{f}                                                             (3-3)

        结合取模运算的性质进行整理计算有:

(i+1)\%n=(2i+1)\%n                                                   (3-4)

                                              \Leftrightarrow [(2i+1)\%n-(i+1)\%n ]\%n = 0\%n

                                              \Leftrightarrow i\%n = 0

        即当步数 i 为总结点数 n 的整数倍时两指针标号相等:

i=kn     (k=0,1,2\cdots )                                                  (3-5)

        又因为对于一个链表,结点数 n 为一个有限值,总是能够找到这样的正整数 i ,使得式3-5成立,故而能够使得两指针的标号相等,即两个指针必然会相遇。

        此时根据式3-5和式3-1求得相遇点的标号为:

I_{s}=I_{f}=1                                                                      (3-6)

        即两指针总是在起始点相遇,\!\!证毕。

4、问题的一般性分析

        上面我们只是讨论了两个指针在同一起始点时的相遇问题,但事实上这还远远不够,因为两个指针不能够保证同时到达环的起点,如图4所示,当slow指针到达环内时,有:

   slow\rightarrow 4                                      fast\rightarrow 5   

 图4:示意图

        对于该问题先进行数学描述,分析从slow指针进入链表环起之后能否实现两个指针相遇。

  图5:两指针初始位置和移动 i 步后的位置

        若选择 fast 指针的初始位置 I_{f0} 为参考点I_{base},该结点所在的位置标号为1,可以认为 slow 指针的初始位置 I_{s0}为参考点的位置的基础上多走了 j 步,即:

I_{base}=I_{f0}=1                                                                            (4-1)

I_{s0}=I_{base}+j         其中 (0\leqslant j< n)                                       (4-2)

        针对 slow 指针,其速度为一步一格,初始值为  1+j,标号建立如式4-3:

                                            I_{s}=(i+j+1)\% n                                                            (4-3)

        针对fast指针,其速度为一步两格,初始值为1,标号建立如式4-4:

 I_{f}=(2i+1)\% n                                                                  (4-4)

        当标号相同时,和上面同样的整理有:

(i-j)\%n=0                                                                       (4-4)

i=kn+j        (k=0,1,2\cdots )                                          (4-5)

        即当步数i与j的差为结点数n的整数倍时成立。

        此时根据式4-5和式4-3求得相遇点的标号为:

I_{s}=I_{f}=(2j+1)\%n                                                        (4-6) 

        可见,两个指针的从同一起始点出发是当 j=0 时的特殊情况。

三、总结

        此处分析是我个人对龟兔赛跑算法的理解,上述证明过程可能有不严谨之处,欢迎各位小伙伴积极讨论指正!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值