1664. Ways to Make a Fair Array - Detailed Explanation
Problem Statement
Description:
You are given an integer array nums
. An array is called fair if the sum of the elements at even indices is equal to the sum of the elements at odd indices. Your task is to determine how many ways you can remove exactly one element from the array so that the resulting array becomes fair. Note: When you remove an element, the indices of the remaining elements change (i.e., elements to the right shift left by one).
Constraints:
- 1 ≤
nums.length
≤ 10<sup>5</sup> - 1 ≤
nums[i]
≤ 10<sup>4</sup>
Example 1:
- Input:
nums = [2, 1, 6, 4]
- Output:
1
- Explanation:
- If you remove the element at index 1 (value
1
), the array becomes[2, 6, 4]
. - New indices: 0 → 2, 1 → 6, 2 → 4
- Even-index sum =
2 + 4 = 6
, odd-index sum =6
- Since both sums are equal, this is a fair array.
- If you remove the element at index 1 (value
Example 2:
- Input:
nums = [1, 1, 1]
- Output:
3
- Explanation:
- Removing any element results in a 2-element array where the first element is at index 0 and the second at index 1.
- For every removal:
- New array:
[1, 1]
- Even-index sum =
1
, odd-index sum =1
- New array:
- All three removal choices yield a fair array.
Example 3:
- Input:
nums = [1, 2, 3]
- Output:
0
- Explanation:
- No matter which element you remove, the sum of even-indexed elements will not equal the sum of odd-indexed ones.
- Therefore, there are no valid removals that make the array fair.
Hints
- Hint 1: Consider how the indices change when you remove an element. The elements to the right of the removed element will shift one position to the left, changing their parity (even/odd index positions).
- Hint 2: Precompute the total sums of even and odd indexed elements. Then, as you iterate through the array, maintain cumulative sums (prefix sums) for the left side of the removal index.
- Hint 3: Use these cumulative sums to efficiently compute what the new even and odd sums would be if you remove the current element without recomputing the sum for the entire array each time.
Approaches
Brute Force Approach
Idea:
For each index in the array, remove the element and then recompute the sums for the even and odd positions for the resulting array. Count the removals that yield a fair array.
Steps:
- Loop through every index
i
innums
. - Create a new array without
nums[i]
. - Recompute the sum of elements at even indices and at odd indices.
- Check if the two sums are equal; if so, increment a counter.
Complexity:
- Time Complexity: O(n²) in the worst case (because for each element, you scan the remaining array).
- Space Complexity: O(n) for the new array each time (if not done in place).
Visual Example:
For nums = [2, 1, 6, 4]
:
- Remove index 0 → New array:
[1, 6, 4]
→ even sum =1 + 4 = 5
, odd sum =6
- Remove index 1 → New array:
[2, 6, 4]
→ even sum =2 + 4 = 6
, odd sum =6
(fair) - Remove index 2 → New array:
[2, 1, 4]
→ even sum =2 + 4 = 6
, odd sum =1
- Remove index 3 → New array:
[2, 1, 6]
→ even sum =2 + 6 = 8
, odd sum =1
Only one removal yields a fair array.
Optimal Approach Using Prefix Sums
Idea:
Rather than simulating each removal, use prefix sums to track the cumulative sums of even-indexed and odd-indexed elements.
-
First, compute the total sum of even indices (
totalEven
) and odd indices (totalOdd
) for the entire array. -
Then, iterate through the array while maintaining cumulative sums for the elements to the left (
leftEven
,leftOdd
). -
For each index
i
, the right part of the array (elements afteri
) will shift and the parity will flip:- If
i
is even:- Remove
nums[i]
fromtotalEven
(because it is being removed). - New even sum becomes:
leftEven + (totalOdd - leftOdd)
- New odd sum becomes:
leftOdd + (totalEven - leftEven)
- Remove
- If
i
is odd:- Remove
nums[i]
fromtotalOdd
. - New even sum becomes:
leftEven + (totalOdd - leftOdd)
- New odd sum becomes:
leftOdd + (totalEven - leftEven)
- Remove
- Note: Although the formulas look similar, the removal step is critical (subtracting
nums[i]
from the respective total).
- If
-
Check if the new even and odd sums are equal; if so, increment your counter.
-
Update the left cumulative sums with the current element (depending on its original index parity).
Complexity:
- Time Complexity: O(n) – a single pass to compute totals, then one pass to check each removal.
- Space Complexity: O(1) – using only a few extra variables.
Step-by-Step Walkthrough (Visual Example):
Consider nums = [2, 1, 6, 4]
-
Initialization:
totalEven = 2 + 6 = 8
totalOdd = 1 + 4 = 5
leftEven = 0
,leftOdd = 0
- Answer counter = 0
-
Iteration:
- Index 0 (value 2, even index):
- Remove 2 → Adjusted
totalEven
becomes8 - 2 = 6
- New even sum =
leftEven (0) + (totalOdd - leftOdd = 5 - 0) = 5
- New odd sum =
leftOdd (0) + (totalEven (after removal) - leftEven = 6 - 0) = 6
- Not fair.
- Update
leftEven = 0 + 2 = 2
- Remove 2 → Adjusted
- Index 1 (value 1, odd index):
- Remove 1 → Adjusted
totalOdd
becomes5 - 1 = 4
- New even sum =
leftEven (2) + (totalOdd - leftOdd = 4 - 0) = 6
- New odd sum =
leftOdd (0) + (totalEven - leftEven = 6 - 2) = 4
- Not fair.
- Update
leftOdd = 0 + 1 = 1
- Remove 1 → Adjusted
- Index 2 (value 6, even index):
- Remove 6 → Adjusted
totalEven
becomes6 - 6 = 0
- New even sum =
leftEven (2) + (totalOdd - leftOdd = 4 - 1) = 2 + 3 = 5
- New odd sum =
leftOdd (1) + (totalEven - leftEven = 0 - 2) = 1 - 2 = -1
- Not fair.
- Update
leftEven = 2 + 6 = 8
- Remove 6 → Adjusted
- Index 3 (value 4, odd index):
- Remove 4 → Adjusted
totalOdd
becomes4 - 4 = 0
- New even sum =
leftEven (8) + (totalOdd - leftOdd = 0 - 1) = 8 - 1 = 7
- New odd sum =
leftOdd (1) + (totalEven - leftEven = 0 - 8) = 1 - 8 = -7
- Not fair.
- Remove 4 → Adjusted
- Index 0 (value 2, even index):
In this particular example, notice that a different perspective on cumulative sums may be needed. (For the sample input [2,1,6,4] the fair removal is at index 1 when computed carefully with the parity shift.)
Important: The parity flip for the right segment means the formulas are applied exactly as:
- If
i
is even:- New even =
leftEven + (totalOdd - leftOdd)
- New odd =
leftOdd + (totalEven - leftEven - nums[i])
- New even =
- If
i
is odd:- New even =
leftEven + (totalOdd - leftOdd - nums[i])
- New odd =
leftOdd + (totalEven - leftEven)
- New even =
(When following the iterative update, be sure to subtract nums[i]
before computing the new sums.)
For example, re-evaluating index 1 in [2,1,6,4]:
- Index 1 (value 1, odd):
- Before removal,
totalOdd = 5
. Removenums[1]
→ NewtotalOdd
becomes5 - 1 = 4
. - New even =
leftEven (2) + (totalOdd - leftOdd) = 2 + (4 - 0) = 6
- New odd =
leftOdd (0) + (totalEven - leftEven) = 0 + (8 - 2) = 6
- Fair removal found. (Counter becomes 1)
- Before removal,
Code Implementation
Python Code
Java Code
Complexity Analysis
-
Brute Force Approach:
-
Time: O(n²)
-
Space: O(n) per removal (if copying the array)
-
-
Optimal Approach:
-
Time: O(n) – single pass for totals and one pass for checking each removal
-
Space: O(1) – only a few extra variables are used
-
Detailed Walkthrough and Visual Example
Let’s walk through the optimal approach with the array [2, 1, 6, 4]
:
-
Initialization:
- Compute totals:
totalEven = 2 (index 0) + 6 (index 2) = 8
totalOdd = 1 (index 1) + 4 (index 3) = 5
- Set
leftEven = 0
andleftOdd = 0
.
- Compute totals:
-
Iteration:
- At index 0 (value 2, even):
- Remove
2
:totalEven
becomes8 - 2 = 6
. - New sums:
- Even =
leftEven (0)
+totalOdd (5)
= 5 - Odd =
leftOdd (0)
+totalEven (6)
= 6
- Even =
- Not fair.
- Update:
leftEven
becomes0 + 2 = 2
.
- Remove
- At index 1 (value 1, odd):
- Remove
1
:totalOdd
becomes5 - 1 = 4
. - New sums:
- Even =
leftEven (2)
+totalOdd (4)
= 6 - Odd =
leftOdd (0)
+totalEven (6)
= 6
- Even =
- Fair removal found.
- Update:
leftOdd
becomes0 + 1 = 1
.
- Remove
- At index 2 (value 6, even):
- Remove
6
:totalEven
becomes6 - 6 = 0
. - New sums:
- Even =
leftEven (2)
+totalOdd (4)
= 6 - Odd =
leftOdd (1)
+totalEven (0)
= 1
- Even =
- Not fair.
- Update:
leftEven
becomes2 + 6 = 8
.
- Remove
- At index 3 (value 4, odd):
- Remove
4
:totalOdd
becomes4 - 4 = 0
. - New sums:
- Even =
leftEven (8)
+totalOdd (0)
= 8 - Odd =
leftOdd (1)
+totalEven (0)
= 1
- Even =
- Not fair.
- Remove
- At index 0 (value 2, even):
-
Final Count:
- Only one index (index 1) yields a fair array.
- Answer = 1
Additional Sections
Common Mistakes
-
Ignoring the Parity Shift:
Many beginners forget that after removal, the indices of the elements to the right shift and their parity (even/odd) flips. -
Incorrect Prefix Sum Update:
Failing to update the cumulative sums correctly before or after the removal check may lead to wrong results. -
Edge Cases:
Not considering cases where the array length is 1 or when all elements are identical.
Edge Cases
-
Single Element Array:
Removing the only element leaves an empty array (which can be considered fair by definition or might be specified as invalid based on problem details). -
Uniform Array:
Arrays where all elements are the same (e.g.,[1, 1, 1, 1]
) might have multiple fair removals.
Alternative Variations and Related Problems
- Variations:
- Removing multiple elements to achieve fairness.
- Given a series of removal queries, determine if the array can be made fair.
Related Problems for Further Practice
Conclusion
This detailed guide covered both a brute force and an optimal approach to solve the "Ways to Make a Fair Array" problem. By using prefix sums and understanding how the parity of indices changes after an element removal, you can achieve an optimal solution with O(n) time complexity. The provided Python and Java implementations, along with the step-by-step walkthrough and discussion of common pitfalls and variations, should equip you with a strong understanding to tackle similar coding interview problems.
GET YOUR FREE
Coding Questions Catalog
