判断单链表中是否存在环及查找环的入口点

出处:http://hi.baidu.com/iwitggwg/item/7ad684119a27fefc9c778a5c

问题1:如何判断单链表中是否存在环(即上图中从结点E到结点R组成的环)?
解答:
        设一快一慢两个指针(Node *fast, *low)同时从链表起点开始遍历,其中快指针每次移动长度为2,慢指针则为1。则若无环,开始遍历之后fast不可能与low重合,且fast或fast->next最终必然到达NULL;若有环,则fast必然不迟于low先进入环,且由于fast移动步长为2,low移动步长为1,则在low进入环后继续绕环遍历一周之前fast必然能与low重合(且必然是第一次重合)。于是函数可写如下:
// 若有环,encounter是fast与low重合的地方
bool hasCircle(Node* head, Node* &encounter)
{
        Node *fast = head, *low = head;
        while(fast && fast->next)
       {
              fast = fast->next->next;
             low = low->next;
             if(fast == low)
              {
                  encounter = fast;
                  return true;
              }
         }
        // fast == NULL || fast->next == NULL
        encounter = NULL;
        return false;
}

问题2:若存在环,如何找到环的入口点(即上图中的结点E)?
解答:
       如图中所示,设链起点到环入口点间的距离为x,环入口点到问题1中fast与low重合点的距离为y,又设在fast与low重合时fast已绕环n周(n>0),且此时low移动总长度为s,则fast移动总长度为2s,环的长度为r。则
s + nr = 2s,n>0      ①
s = x + y                  ②
由①式得
s = nr                 
代入②式得
nr = x + y
x = nr - y                  ③
        现让一指针p1从链表起点处开始遍历,指针p2从encounter处开始遍历,且p1和p2移动步长均为1。则当p1移动x步即到达环的入口点,由③式可知,此时p2也已移动x步即nr - y步。由于p2是从encounter处开始移动,故p2移动nr步是移回到了encounter处,再退y步则是到了环的入口点。也即,当p1移动x步第一次到达环的入口点时,p2也恰好到达了该入口点。于是函数可写如下:
Node* findEntry(Node* head, Node* encounter)
{ 
         Node *p1 = head, *p2 = encounter;
        while(p1 != p2)
        {
               p1 = p1->next;
              p2 = p2->next;
         }
        return p1;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值