689. Maximum Sum of 3 Non-Overlapping Subarrays - Detailed Explanation
Problem Statement
Description:
Given an integer array nums
and an integer k
, find three non-overlapping subarrays of length k
with maximum total sum.
A subarray is a contiguous segment of the array. The three subarrays must not overlap.
Return the starting indices of these three subarrays that yield the maximum total sum. If multiple answers exist, return the lexicographically smallest one.
Examples:
-
Example 1:
- Input:
nums = [1,2,1,2,6,7,5,1]
,k = 2
- Output:
[0, 3, 5]
- Explanation:
The three subarrays are:- Subarray 1 (starting at index 0):
[1,2]
with sum = 3 - Subarray 2 (starting at index 3):
[2,6]
with sum = 8 - Subarray 3 (starting at index 5):
[7,5]
with sum = 12
Total sum = 3 + 8 + 12 = 23, which is the maximum possible.
- Subarray 1 (starting at index 0):
- Input:
-
Example 2:
- Input:
nums = [1,2,1,2,1,2,1,2,1]
,k = 2
- Output:
[0,2,4]
- Explanation:
Although several valid combinations exist, the lexicographically smallest answer is returned.
- Input:
Constraints:
- ( 1 \leq nums.length \leq 2 \times 10^4 )
- ( 1 \leq k \leq \lfloor \text{nums.length} / 3 \rfloor )
- The answer is guaranteed to exist.
Hints
-
Hint 1:
Think about precalculating the sum of every subarray of lengthk
using a sliding window. This will allow you to quickly reference the sum of any subarray. -
Hint 2:
To efficiently find the best left and right subarray candidates for any middle subarray, consider precomputing two arrays:- A left array where for every index
i
you store the starting index of the subarray with the maximum sum in the range ([0, i]). - A right array where for every index
i
you store the starting index of the subarray with the maximum sum in the range ([i, \text{end}]).
- A left array where for every index
Approaches
Approach 1: Brute Force (Not Feasible)
Explanation
-
Idea:
Try every possible triple of non-overlapping subarrays and calculate their total sum. -
Steps:
- Iterate through all possible starting indices for the first subarray.
- For each, iterate for the second subarray ensuring it does not overlap.
- For each pair, iterate for the third subarray ensuring no overlap.
- Track the maximum total sum and the corresponding indices.
-
Complexity:
The time complexity would be O(n³) and is impractical for large inputs.
Approach 2: Sliding Window with Precomputation of Best Candidates
Explanation
This is the optimal and standard approach for this problem.
-
Compute Subarray Sums:
- Use a sliding window to calculate the sum of every contiguous subarray of length
k
. - Store these sums in an array
sum[]
wheresum[i]
is the sum of the subarray starting at indexi
.
- Use a sliding window to calculate the sum of every contiguous subarray of length
-
Precompute the Best Left Subarray for Each Position:
- Create an array
left[]
whereleft[i]
stores the starting index of the subarray (from 0 toi
) that has the maximum sum. - Traverse the
sum[]
array from left to right, updating the best index as you go.
- Create an array
-
Precompute the Best Right Subarray for Each Position:
- Create an array
right[]
whereright[i]
stores the starting index of the subarray (fromi
to end) that has the maximum sum. - Traverse the
sum[]
array from right to left.
- Create an array
-
Iterate for the Middle Subarray:
- For each possible middle subarray (its starting index
j
should be betweenk
andn - 2*k
), combine:- The best left subarray ending before
j
(given byleft[j-1]
). - The middle subarray at index
j
. - The best right subarray starting after
j+k-1
(given byright[j+k]
).
- The best left subarray ending before
- Calculate the total sum and update the maximum if this combination yields a larger sum.
- In case of ties, the lexicographically smallest indices are chosen by design of the left/right arrays.
- For each possible middle subarray (its starting index
Visual Walkthrough
Consider the array from Example 1:
nums = [1,2,1,2,6,7,5,1]
with k = 2
-
Compute Subarray Sums:
- Subarray starting at 0:
[1,2]
→ sum = 3 - Starting at 1:
[2,1]
→ sum = 3 - Starting at 2:
[1,2]
→ sum = 3 - Starting at 3:
[2,6]
→ sum = 8 - Starting at 4:
[6,7]
→ sum = 13 - Starting at 5:
[7,5]
→ sum = 12 - Starting at 6:
[5,1]
→ sum = 6
- Subarray starting at 0:
-
Left Array Computation:
left[0] = 0
- For index 1, best remains 0 (since 3 == 3, choose the earlier index).
- For index 2, best remains 0.
- For index 3, best becomes 3 (8 > 3).
- For index 4, best becomes 4 (13 > 8).
- And so on…
-
Right Array Computation:
right[6] = 6
- For index 5, best becomes 5 (12 > 6).
- For index 4, best remains 4 (13 > 12).
- Continue updating from right to left.
-
Choose Middle Subarray:
- Try every valid
j
for the middle subarray. For each, compute:
total = sum[left[j-1]] + sum[j] + sum[right[j+k]]
- Update the maximum total sum and store the indices.
- Try every valid
Pseudocode
function maxSumOfThreeSubarrays(nums, k):
n = length(nums)
sum = array of size (n-k+1)
window_sum = sum(nums[0:k])
sum[0] = window_sum
for i = 1 to n-k:
window_sum = window_sum + nums[i+k-1] - nums[i-1]
sum[i] = window_sum
left = array of size length(sum)
best = 0
for i from 0 to length(sum)-1:
if sum[i] > sum[best]:
best = i
left[i] = best
right = array of size length(sum)
best = length(sum)-1
for i from length(sum)-1 down to 0:
if sum[i] >= sum[best]:
best = i
right[i] = best
max_total = 0
answer = [-1, -1, -1]
for j from k to length(sum)-k:
i = left[j - k] // best left subarray
l = right[j + k] // best right subarray
total = sum[i] + sum[j] + sum[l]
if total > max_total:
max_total = total
answer = [i, j, l]
return answer
Code Examples
Python Code
Java Code
Complexity Analysis
-
Time Complexity:
- Calculating all subarray sums: O(n)
- Precomputing left and right arrays: O(n) each
- Iterating for the middle subarray: O(n)
- Overall: O(n)
-
Space Complexity:
- Extra space for the
subarraySum
,left
, andright
arrays: O(n) - Overall: O(n)
- Extra space for the
Common Mistakes and Edge Cases
Common Mistakes
-
Overlapping Subarrays:
Not ensuring the three subarrays do not overlap. The indices for the middle subarray must allow enough room for valid left and right subarrays. -
Incorrect Handling of Ties:
When multiple candidates yield the same sum, the lexicographically smallest result is required. Using>=
in the right array calculation ensures that. -
Index Boundaries:
Be cautious when iterating over possible middle subarray indices so that indices for left and right candidates remain valid.
Edge Cases
-
Minimum Array Length:
Whennums.length == 3 * k
, there is only one way to choose the subarrays. -
Uniform Array Values:
When all elements are the same, multiple answers might exist; the lexicographical order requirement will determine the output.
Alternative Variations and Related Problems
Variations
-
Maximum Sum of k Non-Overlapping Subarrays:
Generalize the problem to findk
non-overlapping subarrays. -
Weighted Interval Scheduling:
A related problem where you choose intervals (or subarrays) to maximize the total weight or sum.
Related Problems for Further Practice
-
Best Time to Buy and Sell Stock:
Though different in context, it also involves choosing segments to maximize profit. -
Longest Subarray with Sum Constraint:
Finding a subarray under given constraints is a common theme in array problems. -
Maximum Sum Subarray:
The classic Kadane’s algorithm problem is a stepping stone toward more complex interval selection problems.
GET YOUR FREE
Coding Questions Catalog
