Problem Statement (LeetCode 15 - 3Sum)
Given an integer array nums
, return all the triplets [nums[i], nums[j], nums[k]]
such that:
i != j
,i != k
, andj != k
nums[i] + nums[j] + nums[k] == 0
Notice that the solution set must not contain duplicate triplets.
Example 1:
Input: nums = [-1,0,1,2,-1,-4]
Output: [[-1,-1,2],[-1,0,1]]
Example 2:
Input: nums = []
Output: []
Example 3:
Input: nums = [0]
Output: []
Optimal Approach: Sorting + Two Pointers
Intuition
- Sorting: Sort the input array to allow the use of two pointers to avoid checking for duplicate triplets.
- Iterating with a Fixed Element: Use the first element of the array as the fixed element, and then solve the two-sum problem for the rest of the array using two pointers.
- Two Pointers Approach: For each fixed element, place two pointers on the remaining part of the array and move them accordingly to find pairs that sum up to the target (
0 - nums[i]
).
Why This Approach Works
- Efficiency: Sorting ensures that duplicates can be avoided by skipping over identical elements, and the two-pointer approach reduces the problem from O(n^3) to O(n^2).
- Handling Duplicates: After fixing one element, we skip over duplicate elements by checking the current number against the previous one.
- Time Complexity: The time complexity is dominated by the sorting step, making the overall time complexity O(n^2).
LeetCode-Compatible C++ Code
class Solution {
public:
vector<vector<int>> threeSum(vector<int>& nums) {
vector<vector<int>> result;
sort(nums.begin(), nums.end()); // Sort the array to use two-pointer technique
for (int i = 0; i < nums.size(); ++i) {
// Skip duplicate elements to avoid repeating triplets
if (i > 0 && nums[i] == nums[i - 1]) {
continue;
}
int target = -nums[i]; // We need to find two numbers that sum up to -nums[i]
int left = i + 1, right = nums.size() - 1;
while (left < right) {
int sum = nums[left] + nums[right];
if (sum == target) {
result.push_back({nums[i], nums[left], nums[right]});
// Skip duplicates for left and right
while (left < right && nums[left] == nums[left + 1]) {
++left;
}
while (left < right && nums[right] == nums[right - 1]) {
--right;
}
++left;
--right;
} else if (sum < target) {
++left;
} else {
--right;
}
}
}
return result;
}
};
Step-by-Step Explanation of Code
-
Sorting:
We first sort the input arraynums
. This helps in using the two-pointer technique efficiently and helps in skipping duplicate elements during the iteration. -
Iterating through the Array:
We iterate over the array using indexi
. For eachi
, we set the target sum as-nums[i]
. Our goal is to find two numbers whose sum equals this target. -
Two Pointers:
After fixing the first number, we use two pointers (left
andright
) to find pairs that sum up to the target. The left pointer starts just afteri
, and the right pointer starts from the end of the array. We move the pointers towards each other based on whether their sum is less than, greater than, or equal to the target. -
Skipping Duplicates:
- To avoid duplicate triplets, we skip duplicate numbers by checking if the current number is the same as the previous one.
- After finding a valid triplet, we increment
left
and decrementright
, skipping over any duplicate values at those positions.
-
Return Result:
After processing all elements and finding the triplets, we return theresult
vector.
Dry Run with Example
Input: nums = [-1, 0, 1, 2, -1, -4]
Step 1: Sort the array: [-4, -1, -1, 0, 1, 2]
Step 2: Iterate over each element (i
from 0 to 3).
- For
i = 0
(element-4
):- Target =
4
left = 1
,right = 5
: Sum of-1 + 2 = 1
(moveleft
)left = 2
,right = 5
: Sum of-1 + 2 = 1
(moveleft
)left = 3
,right = 5
: Sum of0 + 2 = 2
(moveleft
)- No valid triplet found.
- Target =
- For
i = 1
(element-1
):- Target =
1
left = 2
,right = 5
: Sum of-1 + 2 = 1
(valid triplet[-1, -1, 2]
)- Skip duplicates and continue until
left = 3
,right = 4
. - Valid triplet
[-1, 0, 1]
.
- Target =
Step 3: Final output: [[-1, -1, 2], [-1, 0, 1]]
Alternative Approaches & Why Not?
Approach | Time Complexity | Space Complexity | Why Not? |
---|---|---|---|
Brute Force (Triple Nested Loop) | O(n^3) | O(1) | This approach checks all triplets, which is inefficient for large input arrays. |
Sorting + Binary Search | O(n^2 log n) | O(1) | This approach is less efficient due to the extra log n factor introduced by binary search. |
Sorting + HashMap | O(n^2) | O(n) | Requires additional space for the HashMap, making it less space-efficient than the two-pointer approach. |
Why Two-Pointer Approach is the Best
- Time Complexity is O(n^2): Sorting the array takes O(n log n), and the two-pointer technique reduces the need for nested loops.
- Space Complexity is O(1): We don't use any extra data structures (except the result), making this solution space-efficient.
- Avoiding Duplicates: Sorting and skipping duplicates during iteration allows us to avoid duplicate triplets efficiently.
Complexity Analysis
-
Time Complexity: O(n^2)
The sorting step takes O(n log n), and the two-pointer approach inside the loop runs in O(n), resulting in a total time complexity of O(n^2). -
Space Complexity: O(1)
The space complexity is constant, as we only store the result and use a few variables for iteration.
Conclusion
✅ Best Approach: Sorting + Two Pointers
✅ Time Complexity: O(n^2)
✅ Space Complexity: O(1)
✅ Optimal Solution for the Problem
Recommended Problems for Practice
🚀 Master 3Sum with the Two-Pointer Approach!