548. Split Array with Equal Sum
Problem Description
In this problem, we are given an array of integers, nums
, and we need to determine if there exists a triplet of indices (i, j, k)
in the array that meet certain conditions. Specifically, the conditions for the indices are:
- Indices
i
,j
, andk
must maintain the sequence0 < i < j < k < n-1
, wheren
is the length of the array. - The sum of elements in four subarrays is equal. The subarrays are:
Subarray 1
: Elements from index0
toi-1
(both inclusive).Subarray 2
: Elements from indexi+1
toj-1
(both inclusive).Subarray 3
: Elements from indexj+1
tok-1
(both inclusive).Subarray 4
: Elements from indexk+1
to the end of the array (indexn-1
).
A subarray (l, r)
represents the sequence of elements in the array nums
from the l
th element to the r
th element, inclusive.
The aim is to return true
if at least one such triplet (i, j, k)
exists, and false
otherwise.
Intuition
The intuition behind the solution is to find a way to efficiently determine if the sum of elements in the given subarrays are equal without recalculating them each time for different values of i
, j
, and k
. To do this, we can make use of prefix sums and a set.
Here is how we could approach the problem:
-
Prefix Sum Computation:
- First, we can calculate a prefix sum array
s
for the givennums
array. The prefix sum array contains the sum of all numbers up to and including thei
th index ofnums
. This helps us calculate the sum of any subarray in constant time.
- First, we can calculate a prefix sum array
-
Determining Equal Subarray Sums:
- To find the correct index
j
, we need to ensure that we have enough space on either side for indicesi
andk
to exist. Therefore, we scan forj
starting from index 3 ton - 3
. Ifj
is too close to the start or end, there can't be four subarrays with equal sums. - For each
j
, we establish two loops:- An inner loop over
i
, which ranges from 1 toj - 1
, looking for subarrays that could be equal in sum to the other subarrays. - We check if the sum of elements from
0
toi-1
is equal to the sum fromi+1
toj-1
. - When we find such an instance, we add the sum to a set
seen
as a potential candidate for the subarray sum. - An outer loop over
k
, which ranges fromj + 2
ton - 1
, looking for subarrays that could match the ones identified byi
. - We check if the sum of elements from
j+1
tok-1
is equal to the sum fromk+1
ton-1
. - If it is and the sum is already in
seen
, there exists at least one combination of i, j, k that satisfies all conditions, thus, we return true.
- An inner loop over
- To find the correct index
-
Checking for Match:
- If we exit both loops without finding such a triplet, then we conclude that no such triplet exists, and we return false.
By implementing this approach, we avoid recalculating the sum for each subarray from scratch, which would otherwise result in a much less efficient solution.
Learn more about Prefix Sum patterns.
Solution Approach
To implement the solution described in the intuition, we make use of a couple of important programming concepts: prefix sums and hash sets. This enables us to have an efficient algorithm that can solve the problem without repeatedly computing the sums of the subarrays for each possible combination of (i, j, k)
triplet.
Hereโs a step-by-step walk-through of the implementation based on the provided solution code:
-
Initialization and Prefix Sum Computation:
- Initialize an array
s
with lengthn + 1
wheren
is the length of the input arraynums
. Thes
array is going to store the prefix sums ofnums
. - Fill the
s
array with prefix sums, wheres[i + 1] = s[i] + nums[i]
. This meanss[i]
holds the total sum from the start ofnums
up to (and including) indexi-1
.
- Initialize an array
-
Finding the Index
j
:- Iterate over possible values of
j
starting from index 3 ton - 3
. The choice of starting from 3 ensures that there is space for at least two elements beforej
and ending atn - 3
ensures there's space for at least two elements afterj
.
- Iterate over possible values of
-
Exploring Potential Values for
i
:- For each
j
, initialize an empty setseen
. This set is used to store sums of subarrays that could potentially match with other subarrays. - Loop through
i
which starts from 1 and goes up toj - 1
, perform the following:- Check if the sum of the subarray
(0, i - 1)
which iss[i]
, is equal to the sum of the subarray(i + 1, j - 1)
which iss[j] - s[i + 1]
. - If they are equal, add the sum to the set
seen
.
- Check if the sum of the subarray
- For each
-
Exploring Potential Values for
k
:- Next, loop through
k
which starts fromj + 2
and ends atn - 1
, perform the following:- Check if the sum of the subarray
(j + 1, k - 1)
which iss[k] - s[j + 1]
, is equal to the sum of the subarray(k + 1, n - 1)
which iss[n] - s[k + 1]
. - If they are equal and the sum (the sum of the subarray
(j + 1, k - 1)
) exists in theseen
set, we have found a triplet(i, j, k)
that satisfies all the conditions. - At this point, return
true
since the required triplet exists.
- Check if the sum of the subarray
- Next, loop through
-
Returning the Result:
- If we finish both loops and haven't returned
true
, we determine that no such triplet exists, and therefore, we returnfalse
.
- If we finish both loops and haven't returned
This approach is efficient because we utilized the pre-computed prefix sum array to quickly access the sum of any subarray in constant time. Additionally, by using the set to keep track of seen sums, we avoid redundant comparisons for each potential k
, making it a much faster algorithm.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Consider the array nums
with the following elements:
1nums = [1, 2, 1, 2, 1, 2, 1]
Let's walk through the solution approach with this example:
-
Initialization and Prefix Sum Computation:
- First, we initialize an array
s
for storing prefix sums withn + 1
elements, wheren
is the length ofnums
. In this case,n
is 7, sos
will have 8 elements. - Then we compute the prefix sums. So the
s
array after prefix sum computation will be[0, 1, 3, 4, 6, 7, 9, 10]
. For instance,s[4] = 6
representing the sum ofnums
from index0
to2
.
- First, we initialize an array
-
Finding the Index
j
:- We iterate over possible values of
j
from 3 to 4 (asn - 3
is 4 for this example).
- We iterate over possible values of
-
Exploring Potential Values for
i
forj = 3
:- We initialize an empty set
seen
forj = 3
. - Loop through
i
which starts from 1 and goes up to2
(j - 1
):- For
i = 1
, we check the sums:s[1] = 1
ands[3] - s[2] = 4 - 3 = 1
. They are equal, so we add1
toseen
.
- For
- We initialize an empty set
-
Exploring Potential Values for
k
forj = 3
:- We loop through
k
which can only be5
since it starts fromj + 2
and ends atn - 1
for thisj
.- For
k = 5
, we check the sums:s[5] - s[4] = 7 - 6 = 1
ands[7] - s[6] = 10 - 9 = 1
. They are equal, and1
is in theseen
set. - Having found a
k
such that the sum fromj+1
tok-1
matches the sum fromk+1
ton-1
and is also inseen
, we have found a valid triplet(i, j, k)
which is(1, 3, 5)
. - We return
true
since we have found the required triplet.
- For
- We loop through
-
Returning the Result:
- In this case, we found at least one valid triplet that satisfies all the conditions. Hence, if we were to implement this example in code, the result would be
true
.
- In this case, we found at least one valid triplet that satisfies all the conditions. Hence, if we were to implement this example in code, the result would be
This example illustrates how the intuition and approach to the problem can be used to efficiently find a triplet in the array that meets the requirements.
Solution Implementation
1from typing import List
2
3class Solution:
4 def splitArray(self, nums: List[int]) -> bool:
5 # Get the length of the input array
6 n = len(nums)
7
8 # Initialize a prefix sum array with an extra position for simplicity
9 prefix_sum = [0] * (n + 1)
10
11 # Compute the prefix sum array where prefix_sum[i] represents
12 # the sum of elements from nums[0] to nums[i-1]
13 for i, num in enumerate(nums):
14 prefix_sum[i + 1] = prefix_sum[i] + num
15
16 # The main loop to check for split positions starting at index 3
17 # and ending at n-4 to ensure there are enough elements on both sides
18 for mid in range(3, n - 3):
19 # Store sums that can be created from the first half of the array
20 seen_sums = set()
21
22 # Check for all possible splits in the first half
23 for left in range(1, mid - 1):
24 # If a valid split is found, add the sum to the set
25 if prefix_sum[left] == prefix_sum[mid] - prefix_sum[left + 1]:
26 seen_sums.add(prefix_sum[left])
27
28 # Check for all possible splits in the second half of the array
29 for right in range(mid + 2, n - 1):
30 # Check if there is a valid split that matches any sum in 'seen_sums'
31 if (prefix_sum[n] - prefix_sum[right + 1] == prefix_sum[right] - prefix_sum[mid + 1]
32 and prefix_sum[n] - prefix_sum[right + 1] in seen_sums):
33 return True
34
35 # If we reach this point, no valid split was found
36 return False
37
1class Solution {
2 public boolean splitArray(int[] nums) {
3 int n = nums.length;
4 // Array to store the prefix sums, one extra element for ease of calculations
5 int[] prefixSums = new int[n + 1];
6
7 // Calculate the prefix sums
8 for (int i = 0; i < n; ++i) {
9 prefixSums[i + 1] = prefixSums[i] + nums[i];
10 }
11
12 // Traverse through the array, starting from index 3 to n - 4
13 // 'j' is the potential middle split point
14 for (int j = 3; j < n - 3; ++j) {
15 Set<Integer> seenSums = new HashSet<>();
16 // First pass to check possible sums from the left subarray
17 for (int i = 1; i < j - 1; ++i) {
18 if (prefixSums[i] == prefixSums[j] - prefixSums[i + 1]) {
19 seenSums.add(prefixSums[i]);
20 }
21 }
22
23 // Second pass to check matching sums from the right subarray
24 for (int k = j + 2; k < n - 1; ++k) {
25 if (prefixSums[n] - prefixSums[k + 1] == prefixSums[k] - prefixSums[j + 1] && seenSums.contains(prefixSums[n] - prefixSums[k + 1])) {
26 return true; // Found a valid split
27 }
28 }
29 }
30 // If no valid split is found
31 return false;
32 }
33}
34
1class Solution {
2public:
3 // Function that determines if the array can be split into four parts
4 // with the same sum, with one element between these parts.
5 bool splitArray(vector<int>& nums) {
6 int n = nums.size();
7 vector<int> prefixSum(n + 1, 0); // Initialize prefix sums array with an additional 0 at the start.
8
9 // Calculate prefix sums for all elements.
10 for (int i = 0; i < n; ++i) {
11 prefixSum[i + 1] = prefixSum[i] + nums[i];
12 }
13
14 // Use a three-pointer approach to find the split points.
15 for (int middle = 3; middle < n - 3; ++middle) { // middle is the middle cut, avoiding the first 2 and last 2 elements.
16 unordered_set<int> seenSums; // Store sums that we've seen which are candidates for the first section.
17
18 // Find all possible sums for the left section.
19 for (int left = 1; left < middle - 1; ++left) {
20 if (prefixSum[left] == prefixSum[middle] - prefixSum[left + 1]) {
21 // If a sum that can be the left section is found, add it to 'seenSums'.
22 seenSums.insert(prefixSum[left]);
23 }
24 }
25
26 // Find if there's a corresponding sum for the right section.
27 for (int right = middle + 2; right < n - 1; ++right) {
28 if (prefixSum[n] - prefixSum[right + 1] == prefixSum[right] - prefixSum[middle + 1]
29 && seenSums.count(prefixSum[n] - prefixSum[right + 1])) {
30 // If the sum for the right section equals one of the left section sums, return true for a successful split.
31 return true;
32 }
33 }
34 }
35 return false; // Return false if no such split is found.
36 }
37};
38
1// Function that determines if the array can be splitted into four parts with the same sum,
2// with one element between these parts.
3function splitArray(nums: number[]): boolean {
4 let n: number = nums.length;
5 let prefixSum: number[] = new Array(n + 1).fill(0); // Initialize prefix sums array with an additional 0 at the start.
6
7 // Calculate prefix sums for all elements.
8 for (let i = 0; i < n; ++i) {
9 prefixSum[i + 1] = prefixSum[i] + nums[i];
10 }
11
12 // Use a three-pointer approach to find the split points.
13 for (let middle = 3; middle < n - 3; ++middle) { // 'middle' is the middle cut, avoiding the first 2 and last 2 elements.
14 let seenSums: Set<number> = new Set<number>(); // Store sums that we've seen which are candidates for the first section.
15
16 // Find all possible sums for the left section
17 for (let left = 1; left < middle - 1; ++left) {
18 if (prefixSum[left] === prefixSum[middle] - prefixSum[left + 1]) {
19 // If a sum that can be the left section is found, add it to 'seenSums'.
20 seenSums.add(prefixSum[left]);
21 }
22 }
23
24 // Find if there's a corresponding sum for the right section.
25 for (let right = middle + 2; right < n - 1; ++right) {
26 if (prefixSum[n] - prefixSum[right + 1] === prefixSum[right] - prefixSum[middle + 1]
27 && seenSums.has(prefixSum[n] - prefixSum[right + 1])) {
28 // If the sum for the right section equals one of the left section sums, return true for a successful split.
29 return true;
30 }
31 }
32 }
33 return false; // Return false if no such split is found.
34}
35
36// Example usage:
37let nums: number[] = [1, 2, 1, 2, 1, 2, 1];
38console.log(splitArray(nums)); // Output should be true or false depending on the array content.
39
Time and Space Complexity
Time Complexity
The given Python function splitArray
is designed to determine if an array can be split into four parts with equal sums. The function uses a prefix sum array s
to efficiently calculate the sums of subarrays.
Looking at the nested loops, the first loop runs (n-6)
times, where j
ranges from 3 to (n-4)
. The second nested loop runs at most (j-2)
times for each j
. The worst-case scenario for the second loop would be when j
is around n/2
, which would yield roughly (n/2)
iterations. Thus, the innermost condition is checked O(n^2)
times.
The third loop can be considered separately and again runs at most (n-j-2)
iterations for each j
. But this time, the check involves a hash set lookup, which is an O(1)
operation on average. In the worst case, this loop will also contribute to O(n^2)
iterations.
Overall, the time complexity of the code is O(n^2)
due to the nested loops.
Space Complexity
The space complexity for this function is determined by the space required for:
- The prefix sum array
s
, which contains(n+1)
integers. This contributesO(n)
space complexity. - The hash set
seen
, which in the worst-case scenario could store up to((n/2)-2)
sums (since only sums beforej
are considered andj
starts at 3). This also contributesO(n)
space complexity.
Therefore, the overall space complexity of the function is O(n)
.
Learn more about how to find time and space complexity quickly using problem constraints.
Consider the classic dynamic programming of longest increasing subsequence:
Find the length of the longest subsequence of a given sequence such that all elements of the subsequence are sorted in increasing order.
For example, the length of LIS for [50, 3, 10, 7, 40, 80]
is 4
and LIS is
[3, 7, 40, 80]
.
What is the recurrence relation?
Recommended Readings
Prefix Sum The prefix sum is an incredibly powerful and straightforward technique Its primary goal is to allow for constant time range sum queries on an array What is Prefix Sum The prefix sum of an array at index i is the sum of all numbers from index 0 to i By
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.