-
-
Notifications
You must be signed in to change notification settings - Fork 5.6k
/
Copy pathCycleDetectionII.js
57 lines (49 loc) · 1.22 KB
/
CycleDetectionII.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
/**
* A LinkedList based solution for finding the starting node of the cycle in a list.
* @returns the node where cycle begins in the linked list. If there is no cycle present, returns null.
* @see https://en.wikipedia.org/wiki/Cycle_detection
* @see https://leetcode.com/problems/linked-list-cycle-ii/
*/
function findCycleStart(head) {
let length = 0
let fast = head
let slow = head
while (fast !== null && fast.next !== null) {
fast = fast.next.next
slow = slow.next
if (fast === slow) {
length = cycleLength(slow)
break
}
}
if (length === 0) {
// If there is no cycle, return null.
return null
}
let ahead = head
let behind = head
// Move slow pointer ahead 'length' of cycle times
while (length > 0) {
ahead = ahead.next
length--
}
// Now move both pointers until they meet - this will be the start of cycle
while (ahead !== behind) {
ahead = ahead.next
behind = behind.next
}
// return the meeting node
return ahead
}
// head is a node on a cycle
function cycleLength(head) {
// How long until we visit head again?
let cur = head
let len = 0
do {
cur = cur.next
len++
} while (cur != head)
return len
}
export { findCycleStart }