Problem Statement (LeetCode 19)
Given the head of a linked list, remove the nth node from the end of the list and return its head.
Example 1:
Input: head = [1,2,3,4,5], n = 2
Output: [1,2,3,5]
Example 2:
Input: head = [1], n = 1
Output: []
Example 3:
Input: head = [1,2], n = 1
Output: [1]
Optimal Approach: Two Pointers (Fast and Slow Pointers)
Intuition
To solve this problem efficiently, we can use the two pointers technique:
- Fast Pointer: Move the fast pointer
n
steps ahead. - Slow Pointer: After moving the fast pointer, move both fast and slow pointers simultaneously until the fast pointer reaches the end of the list. The slow pointer will be just before the node to be deleted.
By using the two pointers, we avoid having to traverse the list multiple times or storing the nodes, which makes the solution both time-efficient and space-efficient.
LeetCode-Compatible C++ Code
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode() : val(0), next(nullptr) {}
* ListNode(int x) : val(x), next(nullptr) {}
* ListNode(int x, ListNode *next) : val(x), next(next) {}
* };
*/
class Solution {
public:
ListNode* removeNthFromEnd(ListNode* head, int n) {
// Create a dummy node to handle edge cases like removing the head node
ListNode* dummy = new ListNode(0);
dummy->next = head;
ListNode* fast = dummy;
ListNode* slow = dummy;
// Move the fast pointer n steps ahead
for (int i = 0; i <= n; ++i) {
fast = fast->next;
}
// Move both fast and slow pointers until fast reaches the end
while (fast != nullptr) {
fast = fast->next;
slow = slow->next;
}
// Now slow is at the node just before the node to be deleted
slow->next = slow->next->next;
// Return the new head
return dummy->next;
}
};
Step-by-Step Explanation of Code
-
Dummy Node:
A dummy node is created and points to the head of the list. This helps in handling edge cases such as removing the head node without special treatment. -
Fast and Slow Pointers:
Both thefast
andslow
pointers start at the dummy node. Thefast
pointer is movedn
steps ahead of theslow
pointer. -
Moving Both Pointers:
After moving thefast
pointern
steps ahead, both thefast
andslow
pointers are moved one step at a time until thefast
pointer reaches the end of the list. -
Remove the Node:
Once thefast
pointer reaches the end, theslow
pointer is just before the node that needs to be removed. Theslow->next
pointer is updated to skip the node to be removed, effectively removing it from the list. -
Return the New Head:
Finally, the new head of the list (which might be different if the head node was removed) is returned by accessingdummy->next
.
Dry Run with Example
Input: head = [1, 2, 3, 4, 5], n = 2
-
Initialization:
The dummy node is created:dummy -> 1 -> 2 -> 3 -> 4 -> 5
.
fast
andslow
both point todummy
. -
Move
fast
Pointer:
Movefast
pointern
steps ahead (2 steps):
fast
points to node3
(after moving two steps). -
Move Both Pointers:
Move bothfast
andslow
one step at a time:fast
moves to4
,slow
moves to1
.fast
moves to5
,slow
moves to2
.fast
moves tonullptr
,slow
moves to3
.
-
Remove the Node:
slow
is now pointing to node3
. We removeslow->next
, which is node4
. -
Return Result:
The new head of the list is1 -> 2 -> 3 -> 5
.
Output: [1, 2, 3, 5]
Alternative Approaches & Why Not?
Approach | Time Complexity | Space Complexity | Why Not? |
---|---|---|---|
Brute Force (Traverse List Twice) | O(n) | O(1) | Requires two passes over the list. |
Sorting | O(n log n) | O(1) | Sorting is not appropriate for linked lists and doesn't preserve original order. |
Dynamic Programming (DP) | O(n) | O(n) | Overkill for this problem since we don't need to store previous states. |
Two Pointers (Optimal) | O(n) | O(1) | Most efficient and optimal solution for this problem. |
Why Two Pointers is the Best
- Time Complexity: O(n), where
n
is the number of nodes in the linked list. We traverse the list once. - Space Complexity: O(1), since we only use a constant amount of extra space for the pointers.
- Optimal Solution: This approach minimizes time complexity and avoids the need for extra space.
Complexity Analysis
-
Time Complexity: O(n)
We only need to traverse the linked list once using the two pointers. -
Space Complexity: O(1)
We use only a constant amount of space, aside from the input and output.
Conclusion
✅ Best Approach: Two Pointers
✅ Time Complexity: O(n)
✅ Space Complexity: O(1)
✅ Optimal Solution for Linked List Manipulation
Recommended Problems for Practice
- Remove Linked List Elements (LeetCode 203)
- Reverse Linked List (LeetCode 206)
- Merge Two Sorted Lists (LeetCode 21)
🚀 Master the Two Pointers Technique for Efficient Linked List Operations!