977. Squares of a Sorted Array - Detailed Explanation
Problem Statement
Description:
Given an array of integers nums
sorted in non-decreasing order, return an array of the squares of each number, also sorted in non-decreasing order.
Constraints:
- 1 ≤
nums.length
≤ 10⁴ - -10⁴ ≤
nums[i]
≤ 10⁴ - The array
nums
is sorted in non-decreasing order.
Example 1:
- Input:
nums = [-4, -1, 0, 3, 10]
- Output:
[0, 1, 9, 16, 100]
- Explanation:
Squaring the numbers gives[16, 1, 0, 9, 100]
. After sorting, the array becomes[0, 1, 9, 16, 100]
.
Example 2:
- Input:
nums = [-7, -3, 2, 3, 11]
- Output:
[4, 9, 9, 49, 121]
- Explanation:
Squaring the numbers gives[49, 9, 4, 9, 121]
. Sorting them results in[4, 9, 9, 49, 121]
.
Hints to Approach the Problem
- Hint 1: The input array is already sorted; however, after squaring, the order might not be preserved because negative numbers become positive.
- Hint 2: Notice that the largest square will come from the element with the greatest absolute value, which lies at either end of the array.
- Hint 3: Use a two-pointer technique—one pointer starting at the beginning and one at the end—to compare absolute values and place the larger square at the end of a result array, working backward.
Approaches
Brute Force Approach
Idea:
Square every element and then sort the squared values.
Steps:
- Iterate over each element in the array, square it, and store it in a new array.
- Sort the new array.
- Return the sorted array.
Complexity:
- Time Complexity: O(n log n) due to the sorting step.
- Space Complexity: O(n) for storing the squared values.
Optimal Approach Using Two Pointers
Idea:
Since the array is sorted by value (but may include negatives), the largest squared value comes from either end. Use two pointers—one at the start and one at the end—to compare the absolute values. Place the larger square at the end of the result array and move the corresponding pointer.
Steps:
- Initialize two pointers:
left
at index 0 andright
at the last index. - Create an output array of the same length as
nums
. - Set an index
pos
at the last position of the output array. - While
left
≤right
:- Compare
abs(nums[left])
andabs(nums[right])
. - If
abs(nums[left])
is greater, squarenums[left]
and place it atoutput[pos]
, then incrementleft
. - Otherwise, square
nums[right]
and place it atoutput[pos]
, then decrementright
. - Decrement
pos
.
- Compare
- Return the output array.
Complexity:
- Time Complexity: O(n) since we make a single pass through the array.
- Space Complexity: O(n) for the output array.
Detailed Walkthrough (Using the Two-Pointer Approach)
Consider the array: nums = [-4, -1, 0, 3, 10]
-
Initialization:
- Set
left = 0
andright = 4
. - Create an output array:
result = [0, 0, 0, 0, 0]
. - Set
pos = 4
(the last index).
- Set
-
Iteration:
-
Step 1:
- Compare:
abs(nums[left]) = 4
vs.abs(nums[right]) = 10
. - Since 10 is larger, square
10
→100
and place it atresult[4]
. - Update:
right
becomes3
,pos
becomes3
. - Output:
[0, 0, 0, 0, 100]
.
- Compare:
-
Step 2:
- Now,
left = 0
(value-4
),right = 3
(value3
). - Compare:
abs(-4) = 4
vs.abs(3) = 3
. - Since 4 is larger, square
-4
→16
and place it atresult[3]
. - Update:
left
becomes1
,pos
becomes2
. - Output:
[0, 0, 0, 16, 100]
.
- Now,
-
Step 3:
- Now,
left = 1
(value-1
),right = 3
(value3
). - Compare:
abs(-1) = 1
vs.abs(3) = 3
. - Since 3 is larger, square
3
→9
and place it atresult[2]
. - Update:
right
becomes2
,pos
becomes1
. - Output:
[0, 0, 9, 16, 100]
.
- Now,
-
Step 4:
- Now,
left = 1
(value-1
),right = 2
(value0
). - Compare:
abs(-1) = 1
vs.abs(0) = 0
. - Since 1 is larger, square
-1
→1
and place it atresult[1]
. - Update:
left
becomes2
,pos
becomes0
. - Output:
[0, 1, 9, 16, 100]
.
- Now,
-
Step 5:
- Now,
left = 2
andright = 2
(both point to0
). - Square
0
→0
and place it atresult[0]
. - Output:
[0, 1, 9, 16, 100]
.
- Now,
-
-
Conclusion:
The final output array is[0, 1, 9, 16, 100]
.
Code Implementation
Python Code
Java Code
Complexity Analysis
-
Brute Force Approach:
- Time Complexity: O(n log n) due to the sorting step.
- Space Complexity: O(n) for the intermediate squared array.
-
Optimal Two-Pointer Approach:
- Time Complexity: O(n) because we traverse the array only once.
- Space Complexity: O(n) for the output array.
Additional Sections
Common Mistakes
-
Not Preserving Order:
Simply squaring the numbers without reordering them can lead to an unsorted array. -
Pointer Mismanagement:
Errors in updating the left/right pointers or the position index in the result array can cause incorrect results.
Edge Cases
-
All Non-negative or All Non-positive Elements:
When all elements are non-negative, squaring preserves the order. For all non-positive numbers, the resulting squared values may appear in reverse order, and the two-pointer technique correctly handles this. -
Single Element Array:
A single element array should simply return the square of that element.
Alternative Variations and Related Problems
- Variations:
- Modify the problem to return the squared values in descending order.
- Solve similar problems where elements are transformed (e.g., taking absolute values) and need to be sorted without an extra sort step.
- Related Problems for Further Practice:
-
Merge Sorted Array: Merging two sorted arrays in linear time.
-
Two Sum II - Input Array Is Sorted: Using two pointers on a sorted array.
-
Squares of a Sorted Array Variants: Other problems involving transforming and reordering arrays.
-
Conclusion
The Squares of a Sorted Array problem challenges you to combine array manipulation with an efficient traversal strategy. By recognizing that the largest square comes from the extremes of the array, the optimal two-pointer approach allows you to produce the sorted squared array in a single pass, achieving O(n) time complexity. The detailed walkthrough, code implementations in Python and Java, and the discussion of common pitfalls and edge cases should equip you with a robust understanding to tackle this problem during your coding interviews.
GET YOUR FREE
Coding Questions Catalog
