Difficulty | Source | Tags | ||||
---|---|---|---|---|---|---|
Hard |
160 Days of Problem Solving |
|
The problem can be found at the following link: Problem Link
You are given an array arr[]
of size n
, where arr[i]
represents the height of blocks, and the width of each block is 1. Your task is to calculate how much water can be trapped between the blocks after rainfall.
Input:
arr[] = [3, 0, 1, 0, 4, 0, 2]
Output:
10
Explanation:
Water trapped = 0 + 3 + 2 + 3 + 0 + 2 + 0 = 10 units
Input:
arr[] = [3, 0, 2, 0, 4]
Output:
7
Explanation:
Water trapped = 0 + 3 + 1 + 3 + 0 = 7 units
Input:
arr[] = [2, 1, 5, 3, 1, 0, 4]
Output:
9
Explanation:
Water trapped = 0 + 1 + 0 + 1 + 3 + 4 + 0 = 9 units
Input:
arr[] = [1, 2, 3, 4]
Output:
0
Explanation:
No water can be trapped as there are no height bounds on both sides.
$1 < arr.size() < 10^5$ $0 < arr[i] < 10^3$
The problem can be solved efficiently using the two-pointer technique. This avoids the need for additional arrays to store left and right maximum heights, making the solution both time and space efficient. The key idea is to calculate the water trapped at each index by determining the minimum of the maximum heights on the left and right sides.
- Initialize two pointers:
l
(left) starting at index 0 andr
(right) starting at the last index. - Maintain two variables,
lMax
andrMax
, to track the maximum heights seen so far from the left and right sides, respectively. - Use the following rules:
- If
arr[l]
is smaller, calculate the water trapped atl
usinglMax
, updatelMax
if needed, and move the left pointer (l
) one step right. - If
arr[r]
is smaller, calculate the water trapped atr
usingrMax
, updaterMax
if needed, and move the right pointer (r
) one step left.
- If
- Continue until the two pointers meet.
- The total water trapped is the sum of all individual contributions.
- Expected Time Complexity: O(n), as each element in the array is processed exactly once using the two-pointer approach.
- Expected Auxiliary Space Complexity: O(1), as no additional arrays or data structures are used, and only a constant amount of extra space is required for the pointers and variables.
class Solution {
public:
int maxWater(vector<int>& arr) {
int l = 0, r = arr.size() - 1, res = 0, lMax = 0, rMax = 0;
while (l < r) {
arr[l] < arr[r] ? (lMax = max(lMax, arr[l]), res += lMax - arr[l++]) : (rMax = max(rMax, arr[r]), res += rMax - arr[r--]);
}
return res;
}
};
class Solution {
public int maxWater(int[] arr) {
int l = 0, r = arr.length - 1, res = 0, lMax = 0, rMax = 0;
while (l < r) {
if (arr[l] < arr[r]) {
lMax = lMax > arr[l] ? lMax : arr[l];
res += lMax - arr[l++];
} else {
rMax = rMax > arr[r] ? rMax : arr[r];
res += rMax - arr[r--];
}
}
return res;
}
}
class Solution:
def maxWater(self, arr):
l, r, res, lMax, rMax = 0, len(arr) - 1, 0, 0, 0
while l < r:
if arr[l] < arr[r]: res += max(0, lMax - arr[l]); lMax = max(lMax, arr[l]); l += 1
else: res += max(0, rMax - arr[r]); rMax = max(rMax, arr[r]); r -= 1
return res
For discussions, questions, or doubts related to this solution, feel free to connect on LinkedIn: Any Questions. Let’s make this learning journey more collaborative!
⭐ If you find this helpful, please give this repository a star! ⭐