题目描述:

一个链表中包含环,请找出该链表的环的入口结点。

分析:

设置两个指针p1,p2,

两个指针都从链表的头部开始走,不过p1每次走一步,p2每次走两步。

直到相遇的时候,p2走的长度是p1的两倍。

此时让p2从头开始走,p1继续往下走,不过此时两指针都是一步一步走。

再次相遇的地点就是环的入口。

证明:

假设结点数一共有m个,环中的结点数有n个。

第一次相遇的时候,它们肯定是在环中相遇的,p1走了s1步,p2走了2*s1步。

那么此时它们在环中的位置是一样的,即(s1-(m-n))%n=(2*s1-(m-n))%n,

也就是s1-(m-n)和2*s1-(m-n)同余,充要条件是(2*s1-(m-n))-(s1-(m-n))是n的整数倍,即s1是n的整数倍。

此时如果让p1再走(m-n)步,那么p1将停在环的入口处,因为p1一共走了m+n*k的步数。

那么我们此时让p2从起点开始一步一步走,到达环的入口也是(m-n)步。所以此时它们会在环的入口处相遇。

代码:

 /*
struct ListNode {
int val;
struct ListNode *next;
ListNode(int x) :
val(x), next(NULL) {
}
};
*/
class Solution {
public:
ListNode* EntryNodeOfLoop(ListNode* pHead) {
ListNode* p1 = pHead;
ListNode* p2 = pHead;
while(p2 && p2->next) {
p1 = p1->next; // p1走一步
p2 = p2->next->next; // p2走两步
if(p1 == p2) { // 相遇的时候,p2的步数是p1的两倍
p2 = pHead; // 让p1又从头开始走
while(p1 != p2) { // 现在p1和p2都一步一步走,直到他们相遇,相遇的位置就是环的入口
p1 = p1->next;
p2 = p2->next;
}
return p2;
}
}
return NULL;
}
};

最新文章

  1. openresty 前端开发序
  2. 白皮 Chapter 1
  3. (链接)打印相关_.NET打印小资料
  4. Assets和Raw区别
  5. css巧妙实现分隔线
  6. 阿里巴巴fastJson
  7. 真机调试以及“Could not find Developer Disk Image”问题解决方案
  8. liunx使用技巧
  9. Web后台快速开发框架(.NET Core)
  10. oo第三次总结
  11. JS跨域ajax访问
  12. centos6 利用外部的smpt服务器计划任务发送邮件
  13. Newtonsoft.Json反序列化(Deserialize)出错:Bad JSON escape sequence
  14. python之路(二)-collections系列
  15. Spring Maven 包的依赖
  16. Servlet 知识点 中文乱码的本质与解决
  17. 微信小程序中跳转另一个小程序
  18. MPAndroidChart的具体属性方法
  19. lr几个常用的函数
  20. python自动化之连接数据库

热门文章

  1. python模块之codecs: 自然语言编码转换
  2. python 加密方法总结
  3. 重启oracle方法一二三
  4. 大数据(13) - Spark的安装部署与简单使用
  5. epplus excel数据导出(数据量有点大的情况) Web和Client
  6. [openwrt]网络配置
  7. JavaScript的parseint()函数
  8. powershell---高级函数的介绍
  9. 实际用户ID,有效用户ID,保存的设置用户ID
  10. Hadoop1.2.1 配置文件详解