Problem Statement #

Given the head of a Singly LinkedList, write a function to determine if the LinkedList has a cycle in it or not.

Created with Fabric.js 1.6.0-rc.1 Example: head Following LinkedList has a cycle: Following LinkedList doesn't have a cycle: 2 4 6 8 10 null 1 2 3 4 5 6 head

Try it yourself #

Try solving this question here:

Output

0.502s

LinkedList has cycle: False LinkedList has cycle: False LinkedList has cycle: False

Solution #

Imagine two racers running in a circular racing track. If one racer is faster than the other, the faster racer is bound to catch up and cross the slower racer from behind. We can use this fact to devise an algorithm to determine if a LinkedList has a cycle in it or not.

Imagine we have a slow and a fast pointer to traverse the LinkedList. In each iteration, the slow pointer moves one step and the fast pointer moves two steps. This gives us two conclusions:

  1. If the LinkedList doesn’t have a cycle in it, the fast pointer will reach the end of the LinkedList before the slow pointer to reveal that there is no cycle in the LinkedList.
  2. The slow pointer will never be able to catch up to the fast pointer if there is no cycle in the LinkedList.

If the LinkedList has a cycle, the fast pointer enters the cycle first, followed by the slow pointer. After this, both pointers will keep moving in the cycle infinitely. If at any stage both of these pointers meet, we can conclude that the LinkedList has a cycle in it. Let’s analyze if it is possible for the two pointers to meet. When the fast pointer is approaching the slow pointer from behind we have two possibilities:

  1. The fast pointer is one step behind the slow pointer.
  2. The fast pointer is two steps behind the slow pointer.

All other distances between the fast and slow pointers will reduce to one of these two possibilities. Let’s analyze these scenarios, considering the fast pointer always moves first:

  1. If the fast pointer is one step behind the slow pointer: The fast pointer moves two steps and the slow pointer moves one step, and they both meet.
  2. If the fast pointer is two steps behind the slow pointer: The fast pointer moves two steps and the slow pointer moves one step. After the moves, the fast pointer will be one step behind the slow pointer, which reduces this scenario to the first scenario. This means that the two pointers will meet in the next iteration.

This concludes that the two pointers will definitely meet if the LinkedList has a cycle. A similar analysis can be done where the slow pointer moves first. Here is a visual representation of the above discussion:

Created with Fabric.js 1.6.0-rc.1 1 2 3 4 5 6 fast, slow 1 2 3 4 5 6 1 2 3 4 5 6 fast slow fast slow 1 2 3 4 5 6 slow fast 1 2 3 4 5 6 fast, slow

Code #

Here is what our algorithm will look like:

Output

4.344s

LinkedList has cycle: False LinkedList has cycle: True LinkedList has cycle: True

Time Complexity #

As we have concluded above, once the slow pointer enters the cycle, the fast pointer will meet the slow pointer in the same loop. Therefore, the time complexity of our algorithm will be O(N)O(N) where ā€˜N’ is the total number of nodes in the LinkedList.

Space Complexity #

The algorithm runs in constant space O(1)O(1).


Similar Problems #

Problem 1: Given the head of a LinkedList with a cycle, find the length of the cycle.

Solution: We can use the above solution to find the cycle in the LinkedList. Once the fast and slow pointers meet, we can save the slow pointer and iterate the whole cycle with another pointer until we see the slow pointer again to find the length of the cycle.

Here is what our algorithm will look like:

Output

0.463s

LinkedList cycle length: 4 LinkedList cycle length: 3

Time and Space Complexity: The above algorithm runs in O(N)O(N) time complexity and O(1)O(1) space complexity.

Mark as Completed
←    Back
Introduction
Next    →
Start of LinkedList Cycle (medium)