题目1:判断给定的链表是否存在环;
思路:
使用两个具有不同移动速度的指针。如果链表有环,两个指针就会在环中相遇;如果链表没有环,移动快的指针遇到null就结束。
“相遇”其实是一种追赶,移动快的指针会在环中追赶上移动慢的指针;形象一点,可以设想乌龟和兔子在一个环形轨道上赛跑,那么跑得快的兔子就会追赶上乌龟。
设定:指针slowPtr每次移动一个结点,指针fastPtr每次移动两个结点
注意:代码中的ListNode类,参考数据结构与算法(3)——单链表
/**
* 判断给定的链表是否包含环
* @param headNode 链表的头结点
* @return true:表示链表包含环;false:表示链表不包含环
*/
public static boolean doesLinkedListContainsLoop(ListNode headNode){
// 首先判断链表是否存在
if (headNode == null) {
return false;
}
// 初始化slowPtr和fastPtr指针,使其都指向头结点
ListNode slowPtr = headNode;
ListNode fastPtr = headNode;
// slowPtr指针每次移动一个结点;fastPtr每次移动两个结点
while(slowPtr.getNext() != null && fastPtr.getNext().getNext() != null){
slowPtr = slowPtr.getNext();
fastPtr = fastPtr.getNext().getNext();
if (slowPtr == fastPtr) {
return true;
}
}
return false;
}
题目2:判断给定的链表是否存在环,如果存在,找到环的起始结点;
以下的算法证明,参考自http://www.cnblogs.com/ccdev/archive/2012/09/06/2673618.html
算法描述:
当fastPtr在环内第一次最赶上slowPtr时,slowPtr肯定没有走完链表,而fast已经在环内循环了n圈(1<=n)。
假设slowPtr走了s步,则fastPtr走了2s步(fastPtr的步数还等于s加上在环上多转的n圈),设环长为r,则:
2s= s + nr
s=nr
设整个链表长L,入口环与相遇点距离为x,起点到环入口点的距离为a。
因为s = a + x 所以有 a + x = nr
a+x = (n–1)r + r = (n-1)r + (L-a)
a=(n-1)r + (L–a–x)
(L–a–x)为相遇点到环入口点的距离,由此可知,从链表头到环入口点距离 = (n-1)圈的环长 + 相遇点到环入口点距离
于是我们在链表头和相遇点分别设一个指针,两个指针每次各走一步,两个指针必定相遇,且相遇第一点为环入口点。
也就是说,假设使fastPtr指向链表头结点,使slowPtr指向相遇点,两个指针同时移动,每次移动一个结点,
当fastPtr移动到环开始的结点时,移动的距离就为a;此时slowPtr就移动了(n-1)圈的环长+相遇点到环入口点的距离
两个指针相遇的结点就是环的入口结点。
/**
* 找到链表中环的开始结点即入环口。
* @param headNode 链表的头结点
* @return 环入口结点
*/
public static ListNode findBeginOfLoop(ListNode headNode) {
// 先判断链表中是否有环
boolean loopExists = false;
// 初始化slowPtr和fastPtr指针,使其都指向头结点
ListNode slowPtr = headNode;
ListNode fastPtr = headNode;
while(slowPtr.getNext() != null && fastPtr.getNext().getNext() != null){
slowPtr = slowPtr.getNext();
fastPtr = fastPtr.getNext().getNext();
if (slowPtr == fastPtr) {
System.out.println("找到环的入口");
// 链表中环存在
loopExists = true;
// 跳出循环,一定要写,不然就是死循环
break;
}
}
// 如果环存在
if (loopExists) {
fastPtr = headNode;
// slowPtr指针和fastPtr指针每次移动一个结点
while(slowPtr != fastPtr){
slowPtr = slowPtr.getNext();
fastPtr = fastPtr.getNext();
}
return slowPtr;
}
// 如果不存在
return null;
}
题目2:判断给定的链表是否存在环,如果存在,返回环的长度;
思路:首先找到slowPtr指针和fastPtr指针两个指针的相遇点,然后保持fastPtr指针不动,使slowPtr指针继续移动,每次移动一个结点。
/**
* 如果链表中存在环,求环的长度
* @param headNode 链表的头结点
* @return 环的长度
*/
public static int getLengthOfLoop(ListNode headNode){
int length = 0;
// 先判断链表中是否有环
boolean loopExists = false;
// 初始化slowPtr和fastPtr指针,使其都指向头结点
ListNode slowPtr = headNode;
ListNode fastPtr = headNode;
while(slowPtr.getNext() != null && fastPtr.getNext().getNext() != null){
slowPtr = slowPtr.getNext();
fastPtr = fastPtr.getNext().getNext();
if (slowPtr == fastPtr) {
System.out.println("找到环的入口");
// 链表中环存在
loopExists = true;
// 跳出循环,一定要写,不然就是死循环
break;
}
}
// 如果环存在
if (loopExists) {
do {
slowPtr = slowPtr.getNext();
length++;
} while (slowPtr != fastPtr);
}
// 返回环的长度
return length;
}