2772. Apply Operations to Make All Array Elements Equal to Zero
Problem Description
You are given an array of integers, nums
, which is indexed starting from zero. In addition, you're provided with a positive integer, k
, representing the size of a subarray that you can use in an operation. The operation consists of selecting any k
-sized subarray from nums
and decreasing each element within that subarray by 1
. The goal of the problem is to determine whether it's possible to apply this operation any number of times in such a way that all elements of the array can be made equal to 0
. If it's possible, the function should return true
; otherwise, it should return false
. A subarray mentioned in the problem is defined as a contiguous, non-empty segment of the array.
Intuition
The intuition behind the solution to this problem relies on the idea that if you need to reduce an element to zero, you would need to perform the operation a number of times equal to the value of that element. We can start this process from the leftmost element and move right.
At each step, you check the element's value, taking into account any alterations made by previous operations. If the value at the current index can be reduced to zero, you proceed. If you encounter a negative value or an index where the k
-sized subarray goes out of bounds of the array, it implies that it is not possible to reduce all elements to zero; therefore, you would return false
.
Interestingly, instead of directly manipulating the array after each operation (which would be time-consuming), you utilize a technique involving a difference array. With the help of this difference array, you can represent increments and decrements in the array values at specific intervals effectively. By keeping track of the cumulative changes using a prefix sum approach, you can determine the actual value of each element as you iterate through the original array.
The clever insight here is the use of the difference array to make the operations of decrementing a subarray's elements efficient. As you iterate through the original array, you adjust the difference array accordingly to represent the needed operations. Once the iteration is complete without encountering any issues mentioned earlier, you can confidently return true
, indicating that all elements can be made equal to 0
.
Learn more about Prefix Sum patterns.
Solution Approach
The solution approach involves the use of a difference array to efficiently track the cumulative operations that must be performed on the original array, nums
. This difference array, d
, is initially all zeros and has a length one greater than nums
to accommodate changes that affect elements beyond the end of nums
.
Here's the detailed process:
- We initialize the difference array
d
and a variables
which will hold the sum of difference values as we progress throughnums
. - We iterate through the elements of the
nums
array. For each indexi
:- We accumulate the sum
s
by addingd[i]
to it. This variables
represents the total change made tonums[i]
due to operations on earlier elements because a change to an earlier element may affect all subsequent elements within itsk
-sized subarray. - We apply the accumulated sum
s
tonums[i]
to calculate its modified value after previous operations. - If the modified value of
nums[i]
is already0
, we move to the next element, as no action is required. - If the modified value is less than
0
, or if thek
-sized subarray starting ati
would go out of bounds (i.e.,i + k > n
), we returnfalse
. This check ensures that it's impossible to perform an operation without reducing some element below zero or dealing with a subarray that extends beyond the array's end. - If the value is positive and within bounds, this means we need to perform the operation
nums[i]
times on the subarray starting at indexi
and ending at indexi + k - 1
. Thus, we decreases
bynums[i]
to indicate that we've made a plan to carry out these operations. We also incrementd[i + k]
bynums[i]
, effectively canceling out the operation's effect beginning at indexi + k
. This is where the difference arrayd
shows its utility by allowing us to make adjustments at specific points that automatically alter the sum in between.
- We accumulate the sum
- Once the iteration is complete without returning
false
, it means that we have a valid sequence of operations that can reduce all elements to0
, and we returntrue
.
Through the use of a difference array and prefix sums, the solution avoids repeated iterations over subarrays making it efficient and effective for achieving the task in linear time complexity, O(n), where n is the number of elements in nums
.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's walk through an example to illustrate the solution approach described above. Consider an array nums = [3, 4, 2, 3]
and k = 3
.
-
We initialize the difference array
d
which will have one extra element thannums
, sod = [0, 0, 0, 0, 0]
. We also set a variables = 0
. -
Begin iterating through
nums
:-
For
i = 0
,s = d[0] + s = 0
. The value atnums[0]
after adjustments is3
. Since we have ak
-size subarray (which fits within bounds, asi + k = 3
is not greater thann
), and the value is above0
, we need to perform the operation3
times. So, we decreases
by3
to-3
, and incrementd[3]
by3
, yieldingd = [0, 0, 0, 3, 0]
. This represents our commitment to decrease the values in the subarray starting from index0
to index2
. -
For
i = 1
,s = d[1] + s = -3
. The value atnums[1]
is modified to4 - 3 = 1
. Again, we perform the operation1
time, so we decrements
by1
to-4
, and incrementd[4]
by1
, givingd = [0, 0, 0, 3, 1]
. -
For
i = 2
,s = d[2] + s = -4
. The value atnums[2]
is modified to2 - 4 = -2
. Here we encounter a value less than0
after applying the effect of previous operations, which means it's not possible to reduce the original array to all zeros. Hence, we would returnfalse
.
-
-
In this scenario, we don't get to iterate through the entire
nums
because our solution requires us to returnfalse
once we encounter a modified value below0
. Therefore, it is not possible to make all elements equal to0
with the givenk
-sized subarray operations.
This example clearly shows that if at any point in iterating over the array we come across a value below zero, we conclude that it is not possible to use the given operations to reduce the entire array to zeros and thus return false
. The difference array technique allows us to simulate the operations without repetitively decrementing values in k
-sized chunks, resulting in an efficient approach to solving the problem.
Solution Implementation
1from typing import List
2
3class Solution:
4 def checkArray(self, nums: List[int], k: int) -> bool:
5 # Calculate the length of the array nums
6 array_length = len(nums)
7
8 # Initialize an array to track dynamic modifications
9 dynamic_changes = [0] * (array_length + 1)
10
11 # Initialize a variable to carry the sum of modifications
12 sum_modifications = 0
13
14 # Iterate through nums array
15 for index, value in enumerate(nums):
16 # Update the value with modifications seen so far
17 sum_modifications += dynamic_changes[index]
18 value += sum_modifications
19
20 # If the modified value is zero, continue to next iteration
21 if value == 0:
22 continue
23
24 # Return False if value is negative or index + k exceeds the array bounds
25 if value < 0 or index + k > array_length:
26 return False
27
28 # Update the sum of modifications to negate the excess value
29 sum_modifications -= value
30
31 # Record the necessary change at an index k steps ahead
32 dynamic_changes[index + k] += value
33
34 # If all checks pass, return True
35 return True
36
1class Solution {
2
3 public boolean checkArray(int[] nums, int distance) {
4 int length = nums.length;
5 // Array 'differences' will be used to track the changes to be applied.
6 int[] differences = new int[length + 1];
7
8 // 'sum' is used to keep track of the accumulated value to apply.
9 int sum = 0;
10
11 // Iterate through the array 'nums'.
12 for (int index = 0; index < length; ++index) {
13 // Apply the accumulated changes to the current element.
14 sum += differences[index];
15 nums[index] += sum;
16
17 // If the updated value of nums[index] is zero, continue to the next iteration.
18 if (nums[index] == 0) {
19 continue;
20 }
21
22 // If the updated value of nums[index] is negative,
23 // or index + distance is out of bounds, return false.
24 if (nums[index] < 0 || index + distance > length) {
25 return false;
26 }
27
28 // Decrease 'sum' by the value of nums[index] since it needs to be "cancelled out"
29 // in the future iterations.
30 sum -= nums[index];
31
32 // Schedule a change to be applied 'distance' indices ahead of current index.
33 differences[index + distance] += nums[index];
34 }
35
36 // If the method has not returned false, the conditions were satisfied, therefore return true.
37 return true;
38 }
39}
40
1class Solution {
2public:
3 // Function to check if the array can be transformed to a zero array, with increments of 'k' indices
4 bool checkArray(vector<int>& nums, int k) {
5 int size = nums.size(); // Store the size of nums
6 vector<int> diff(size + 1); // Initialize a difference array with an extra element
7
8 int sum = 0; // Variable to handle the running sum
9
10 // Loop through each element of the nums array
11 for (int i = 0; i < size; ++i) {
12 sum += diff[i]; // Update sum according to the diff array
13 nums[i] += sum; // Modify the current element of nums using sum
14
15 // If the value at nums[i] after modification is 0, no changes are needed
16 if (nums[i] == 0) {
17 continue;
18 }
19
20 // If the current value is negative or we're unable to increment subsequent elements, return false
21 if (nums[i] < 0 || i + k > size) {
22 return false;
23 }
24
25 // Modify the running sum and the diff array at the position i+k
26 sum -= nums[i];
27 diff[i + k] += nums[i];
28 }
29 // If all operations are possible without violating any constraints, return true
30 return true;
31 }
32};
33
1// This function checks if it's possible to convert an array to an array of zeroes,
2// by subtracting a number at a position and also from all elements.
3// `k` positions ahead. It returns true if possible, false otherwise.
4function checkArray(nums: number[], k: number): boolean {
5 // Get the length of the array.
6 const length = nums.length;
7
8 // Initialize a difference array with an extra space for ease of processing.
9 // This helps us to keep track of increments and decrements which will be
10 // propagated along the array.
11 const differenceArray: number[] = Array(length + 1).fill(0);
12
13 // This variable `sum` will keep the running total of elements added
14 // or subtracted because of the difference array.
15 let sum = 0;
16
17 // Iterate over the array elements.
18 for (let index = 0; index < length; ++index) {
19 // Add the current difference to sum.
20 sum += differenceArray[index];
21
22 // Apply the current sum to the current element, simulating propagation.
23 nums[index] += sum;
24
25 // If after adding the sum, the number is zero,
26 // we move on without making any changes.
27 if (nums[index] === 0) {
28 continue;
29 }
30
31 // If the number is negative, or the index surpasses bounds when adding `k`,
32 // we cannot fulfill the condition and return false.
33 if (nums[index] < 0 || index + k > length) {
34 return false;
35 }
36
37 // Since we want to zero out the current number, we would need to
38 // subtract it from the current sum and from each of the next `k`
39 // elements in the array.
40 sum -= nums[index];
41
42 // Increment the difference array `k` positions from the current index
43 // by the value of the current number.
44 differenceArray[index + k] += nums[index];
45 }
46
47 // If the loop finishes, it's possible to convert the array
48 // into an array of zeroes by using the method described, so return true.
49 return true;
50}
51
Time and Space Complexity
Time Complexity
The time complexity of the code is O(n)
where n
is the length of the array nums
. This is because the code consists of a single loop that runs n
times. Inside the loop, there are constant time operations being performed, which do not affect the linear nature of the time complexity.
Space Complexity
The space complexity of the code is also O(n)
because an extra array d
of size n + 1
is created to store the cumulative difference. No other data structures are used that grow with the size of the input, hence the space complexity does not exceed linear bounds.
Learn more about how to find time and space complexity quickly using problem constraints.
What's the output of running the following function using input [30, 20, 10, 100, 33, 12]
?
1def fun(arr: List[int]) -> List[int]:
2 import heapq
3 heapq.heapify(arr)
4 res = []
5 for i in range(3):
6 res.append(heapq.heappop(arr))
7 return res
8
1public static int[] fun(int[] arr) {
2 int[] res = new int[3];
3 PriorityQueue<Integer> heap = new PriorityQueue<>();
4 for (int i = 0; i < arr.length; i++) {
5 heap.add(arr[i]);
6 }
7 for (int i = 0; i < 3; i++) {
8 res[i] = heap.poll();
9 }
10 return res;
11}
12
1class HeapItem {
2 constructor(item, priority = item) {
3 this.item = item;
4 this.priority = priority;
5 }
6}
7
8class MinHeap {
9 constructor() {
10 this.heap = [];
11 }
12
13 push(node) {
14 // insert the new node at the end of the heap array
15 this.heap.push(node);
16 // find the correct position for the new node
17 this.bubble_up();
18 }
19
20 bubble_up() {
21 let index = this.heap.length - 1;
22
23 while (index > 0) {
24 const element = this.heap[index];
25 const parentIndex = Math.floor((index - 1) / 2);
26 const parent = this.heap[parentIndex];
27
28 if (parent.priority <= element.priority) break;
29 // if the parent is bigger than the child then swap the parent and child
30 this.heap[index] = parent;
31 this.heap[parentIndex] = element;
32 index = parentIndex;
33 }
34 }
35
36 pop() {
37 const min = this.heap[0];
38 this.heap[0] = this.heap[this.size() - 1];
39 this.heap.pop();
40 this.bubble_down();
41 return min;
42 }
43
44 bubble_down() {
45 let index = 0;
46 let min = index;
47 const n = this.heap.length;
48
49 while (index < n) {
50 const left = 2 * index + 1;
51 const right = left + 1;
52
53 if (left < n && this.heap[left].priority < this.heap[min].priority) {
54 min = left;
55 }
56 if (right < n && this.heap[right].priority < this.heap[min].priority) {
57 min = right;
58 }
59 if (min === index) break;
60 [this.heap[min], this.heap[index]] = [this.heap[index], this.heap[min]];
61 index = min;
62 }
63 }
64
65 peek() {
66 return this.heap[0];
67 }
68
69 size() {
70 return this.heap.length;
71 }
72}
73
74function fun(arr) {
75 const heap = new MinHeap();
76 for (const x of arr) {
77 heap.push(new HeapItem(x));
78 }
79 const res = [];
80 for (let i = 0; i < 3; i++) {
81 res.push(heap.pop().item);
82 }
83 return res;
84}
85
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.