链表环状检测主要有三种方法:
1、追赶法;如 robinzsy。
2、外部记录法;如improgrammer。
3、内部记录法(打记号);如VivianSnow。
内部标记法和外部标记法其实是一个道理,不过就是辅助变量一个是在链表节点内,一个是借助辅助数组或者hash或者AVL,红黑树,把已经访问过的节点地址存起来,每次访问下一个时候做查询处理.
追赶法,利用最大公倍数原理,用2个游标,对链表进行访问,例如:p1,p2, p1访问每步向前进1个节点,p2则每次向前前进2个节点,如果有环则p1,p2必会相遇,如果p2先遇到了NULL节点,则说明没有环.
关于这个解法最形象的比喻就是在操场当中跑步,速度快的会把速度慢的扣圈可以证明,p2追赶上p1的时候,p1一定还没有走完一遍环路,p2也不会跨越p1多圈才追上我们可以从p2和p1的位置差距来证明,p2一定会赶上p1但是不会跳过p1的因为p2每次走2步,而p1走一步,所以他们之间的差距是一步一步的缩小,4,3,2,1,0 到0的时候就重合了根据这个方式,可以证明,p2每次走三步以上,并不总能加快检测的速度,反而有可能判别不出有环
比如,在环的周长L是偶数的时候,初始p2和p1相差奇数的时候,p2每次走三步,就永远和p1不重合,因为他们之间的差距是: 5, 3 , 1, L-1, L-3
如何找到环路的入口? 是这里要重点说明的内容:
解法如下: 当p2按照每次2步,p1每次一步的方式走,发现p2和p1重合,确定了单向链表有环路了.接下来,让p2回到链表的头部,重新走,每次步长不是走2了,而是走1,那么当p1和p2再次相遇的时候,就是环路的入口了。这点可以证明的:
在p2和p1第一次相遇的时候,假定p1走了n步骤,环路的入口是在p步的时候经过的,那么有
p1走的路径: p+c = n; c为p1和p2相交点,距离环路入口的距离
p2走的路径: p+c+k*L = 2*N; L为环路的周长,k是整数
显然,如果从p+c点开始,p1再走n步骤的话,还可以回到p+c这个点, 同时p2从头开始走的话,经过n不,也会达到p+c这点
显然在这个步骤当中p1和p2只有前p步骤走的路径不同,所以当p1和p2再次重合的时候,必然是在链表的环路入口点上。
分享到:
相关推荐
快慢指针算法是一种用于检测链表中是否存在环的有效方法,也称为“龟兔赛跑”算法。这种方法使用两个指针,一个快指针(每次移动两步),一个慢指针(每次移动一步)。 - **初始化**:将快慢指针都指向链表头结点。...
对于链表环状,我们使用了上述的Floyd判圈法来检测并找到环的入口。对于节点不在链表内的异常,我们在遍历`nodesMap`时检查是否有节点没有`nextId`,以此判断其是否为头节点。如果所有节点都有`nextId`,则说明没有...
本次我们关注的是带环链表,即链表中的某个节点指向了链表的其他部分,形成一个环状结构。这样的数据结构在某些特定问题中非常有用,例如Floyd判断环算法、哈希表的开放寻址法等。 带环链表的基本概念: 1. 节点:...
在LeetCode #141中,我们通过快慢指针的方法来检测链表是否为环状。快指针每次移动两步,慢指针每次移动一步,如果存在环,它们最终会在环内相遇;若不存在环,快指针将先到达链表尾部。 2. **环状链表 II**:...
可以使用Floyd判圈算法(也称为龟兔赛跑算法)来检测环状链表。这个算法使用两个指针,一个快指针每次前进两步,慢指针每次前进一步。如果链表存在环,两个指针最终会在环内相遇。如果遇到环,需要跳出循环并返回`...
- **判断链表是否有环**: `cycleLink`可能提供了检测链表中是否存在环的算法,如Floyd判圈法。 3. **源代码说明.txt**: 这个文件很可能是对所有代码实现的详细解释,包括每段代码的功能、实现思路和使用方法。...
对于环状链表,上述方法会进入无限循环,因此我们需要检测到这种情况并打印错误消息。如果节点不在链表内,我们的`findHeadNode`函数将返回`null`,因为找不到头节点。 5. **测试代码**:最后,我们需要编写测试...
在Java编程中,链表是一种常见的数据结构,用于存储一系列有序元素。在这个问题中,我们需要找到一个特殊类型的链表的头节点。链表中的每个节点包含两个属性:`id`和`nextId`,其中`nextId`表示指向下一个节点的`id`...
对于环状链表,我们需要额外的逻辑来检测环。可以使用 Floyd 判断环的算法,也称作龟兔赛跑算法。设置两个指针,一个慢指针每次移动一步,一个快指针每次移动两步。如果链表有环,快指针最终会追上慢指针。如果遇到...
解决这个问题需要对链表的操作有深入的理解,并能处理链表环状结构和异常情况。 首先,我们需要创建一个`Node`类来表示链表的节点,这个类应包含 id 和 nextId 两个属性,以及构造函数和必要的方法。代码可能如下:...