708. Insert into a Sorted Circular Linked List - Detailed Explanation
Problem Statement:
You are given a node from a circular linked list that is sorted in non-decreasing order. (A circular linked list means the last node’s next pointer points to some node in the list, not necessarily the head.) Write a function to insert a new value insertVal
into the list such that it remains a sorted circular list. The given node can be any single node in the list. If the list is empty (i.e. the given node is null
), create a new single circular list and return that node.
Example Inputs and Outputs:
-
Example 1:
- Input:
- List:
[3 -> 4 -> 1]
(Note: The list is circular; here, 1 points back to 3.) insertVal = 2
- List:
- Output: A circular linked list such as
[3 -> 4 -> 1 -> 2]
(or the equivalent rotation) - Explanation: The new value
2
is inserted between1
and3
(or at the appropriate spot in the sorted order).
- Input:
-
Example 2:
- Input:
- List:
[]
(empty list) insertVal = 1
- List:
- Output: A single-node circular list
[1]
- Explanation: Since the list is empty, create a new node that points to itself.
- Input:
-
Example 3:
- Input:
- List:
[1 -> 1 -> 1]
insertVal = 0
- List:
- Output: A circular linked list containing
[1, 1, 1, 0]
- Explanation: All nodes have the same value, so the new value can be inserted anywhere.
- Input:
Constraints:
- The list is sorted in non-decreasing order.
- The list is circular (i.e. the last node’s next pointer points back to some node in the list).
- The given node can be any node in the list.
- If the list is empty, you must create a new single-node circular list.
Hints for Solving the Problem:
-
Handling an Empty List:
- If the input node is
null
, create a new node, set its next pointer to itself, and return that node.
- If the input node is
-
Traversal in a Circular List:
- Since the list is circular, use a do-while loop (or its equivalent) to traverse the list until you return to the starting node.
-
Finding the Insertion Point:
- Look for the correct position between two nodes (let’s call them
curr
andcurr.next
) where the new value should be inserted. - Case A: If
curr.val <= insertVal <= curr.next.val
, then insert between them. - Case B: If you find a “turning point” (where
curr.val > curr.next.val
), then the list is wrapping from the largest element back to the smallest. In this case, ifinsertVal
is either greater than or equal tocurr.val
(new maximum) or less than or equal tocurr.next.val
(new minimum), then insert there. - Case C: If no proper spot is found after one complete loop (this happens when all nodes have the same value), insert the new node anywhere.
- Look for the correct position between two nodes (let’s call them
Approach: One-Pass Insertion by Traversing the Circular List
Step-by-Step Explanation:
-
Empty List Check:
- If the provided node is
null
, create a new node withinsertVal
, point its next to itself, and return it.
- If the provided node is
-
Initialize Traversal:
- Start from the given node and use a pointer (
curr
) to traverse the list.
- Start from the given node and use a pointer (
-
Search for the Insertion Point:
-
Use a loop (do-while style) to examine each pair of consecutive nodes (
curr
andcurr.next
). -
Normal Case: If
curr.val <= insertVal <= curr.next.val
, then the proper spot is found. -
Edge Case (Wrapping Point): If you encounter a node where
curr.val > curr.next.val
, then the list is wrapping around. In this case, ifinsertVal
is greater than or equal tocurr.val
(i.e., it should be the new maximum) or less than or equal tocurr.next.val
(i.e., it should be the new minimum), then insert betweencurr
andcurr.next
. -
Fallback: If you complete one full loop without finding a valid insertion point (e.g., when all nodes have the same value), simply insert the new node anywhere (typically after the starting node).
-
-
Insertion:
- Create a new node, set its next pointer to
curr.next
, and updatecurr.next
to point to the new node.
- Create a new node, set its next pointer to
-
Return the (possibly unchanged) head of the list.
Python Code:
Java Code:
Complexity Analysis:
- Time Complexity:
- In the worst case, you traverse all nodes in the list once (O(n)), where ( n ) is the number of nodes.
- Space Complexity:
- O(1) extra space (only a few pointers are used for traversal and insertion).
Edge Cases:
-
Empty List:
- When the input list is empty, a new node should be created that points to itself.
-
All Nodes with the Same Value:
- If every node has the same value, no "ideal" insertion point exists; in this case, insert the new node anywhere.
-
Insertion at the Turning Point:
- When the list’s maximum element is immediately followed by its minimum element (e.g., [4, 1, 2, 3]), make sure the insertion logic correctly handles new minimum or maximum values.
Common Mistakes:
-
Infinite Loop:
- Not correctly checking the loop termination condition in a circular list can lead to an infinite loop.
-
Missing the Boundary Condition:
- Forgetting to handle the case where all nodes have the same value or missing the insertion at the turning point.
-
Incorrect Pointer Updates:
- Improperly linking the new node (e.g., not updating both
curr.next
andnewNode.next
) can break the list.
- Improperly linking the new node (e.g., not updating both
Related Problems for Further Practice:
-
Insert into a Sorted Linked List (Non-Circular):
- Practice inserting into a non-circular sorted list.
-
Merge Two Sorted Lists:
- Useful for understanding sorted list manipulations.
-
Rotate List:
- Another problem involving pointer manipulation in linked lists.
GET YOUR FREE
Coding Questions Catalog
