795. Number of Subarrays with Bounded Maximum
Problem Description
Given an array of integers called nums
, and two other integers known as left
and right
, the problem asks us to calculate how many non-empty subarrays (continuous parts of the array) have a maximum element that falls within the range [left, right]
. To clarify, for a subarray to be counted, its largest number must be greater than or equal to left
and less than or equal to right
. Essentially, we're tasked with identifying portions of the array where if you pick out the biggest number, it wouldn't be smaller than left
or larger than right
.
Intuition
To solve this problem, we can use the concept of counting subarrays with elements less than or equal to a given maximum. The provided solution uses a helper function f(x)
which counts subarrays where all elements are less than or equal to x
. The idea is that we can count all subarrays that have elements under or equal to right
, which captures all subarrays that could potentially meet our condition. Then, we subtract the count of subarrays with elements under or equal to left - 1
, which represents subarrays that don't meet our condition (since their maximum would be too small).
The reasoning here is that when we have the count of subarrays with maximums not surpassing right
, we include both valid subarrays (those we are looking for) and subarrays that have maximums that are too small. We don't want the latter, so we find out how many of them there are (using the same count technique but with a maximum of left - 1
) and remove them from our total. The difference gives us exactly the number of valid subarrays we're seeking.
The f(x)
function works by scanning through the array and using a temporary count t
to keep track of valid subarrays ending at the current position. If a value is encountered that is not greater than x
, it can be added to existing valid subarrays to form new ones; hence, t
is incremented by 1. If a value is too large, t
is reset to 0, as no new valid subarrays can end at this position. The variable cnt
accumulates the count of all valid subarrays as we iterate through nums
.
Learn more about Two Pointers patterns.
Solution Approach
The approach taken in the given solution involves a clever use of prefix sums, a common technique in which we construct an array of sums that represent the sum of elements up to a certain index. However, in this specific implementation, rather than directly computing a sum, we are counting the possible subarrays up to a certain index that respect the maximum value condition.
Let's break down the f(x)
function, which is at the heart of the approach:
- Initialize
cnt
andt
to 0.cnt
will hold the total count of subarrays that do not exceed the maximum allowed elementx
, andt
will count such subarrays ending at the current index innums
. - Iterate over each value
v
in thenums
array. - If
v
is greater thanx
, then any subarray includingv
cannot be part of our solution, so we resett
to 0. This essentially says "you can't start or continue a valid subarray from this point." - If
v
is less than or equal tox
, then we can extend every subarray counted byt
with this new valuev
plus one extra: the subarray that only includesv
itself. Thus, we incrementt
by 1. - For every iteration, we add
t
tocnt
which accumulates our count.
The outer function numSubarrayBoundedMax
makes use of f(x)
to apply our subarray counting method twice:
- First, we count all valid subarrays with maximums not more than
right
, which gives usf(right)
. - Then, we count the subarrays that are strictly smaller than
left
, which isf(left - 1)
.
The subarrays we want to exclude from our count are the ones where the maximum element value is strictly less than 'left'. By counting these with f(left - 1)
, we get the count of subarrays with maximum values in the range [0, left-1]
. Thus, when we subtract f(left - 1)
from f(right)
, we remove these invalid subarrays from our total, leaving us with the number of valid subarrays that have their maximum element between left
and right
, inclusive.
This elegant subtraction of prefix counts helps to efficiently compute the final desired count without having to evaluate each subarray independently, thereby reducing the time complexity.
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 a small example to illustrate the solution approach:
Suppose we have the array nums = [2, 1, 4, 3]
and the integers left = 2
and right = 3
. We want to count the number of subarrays where the maximum is at least left
and at most right
.
First, let's consider the f(x)
function that counts the subarrays with maximum elements less than or equal to x
:
-
Using
f(right)
, which isf(3)
:- Initialize
cnt = 0
andt = 0
. - For
v = 2
(first value),t
is incremented to1
(t += 1
), socnt = 1
. - For
v = 1
(second value),t
is incremented to2
(t += 1
), socnt = 3
. - For
v = 4
(third value),t
is reset to0
, as4
is greater than3
. - For
v = 3
(fourth value),t
is incremented to1
(t += 1
), socnt = 4
.
f(3)
gives us a count of4
subarrays. - Initialize
-
Using
f(left - 1)
, which isf(1)
:- Initialize
cnt = 0
andt = 0
. - For
v = 2
(first value),t
is reset to0
, since2
is greater than1
. - For
v = 1
(second value),t
is incremented to1
(t += 1
), socnt = 1
. - For
v = 4
(third value),t
is reset to0
, as it's greater than1
. - For
v = 3
(fourth value),t
is reset again to0
, since3
is also greater than1
.
f(1)
gives us a count of1
subarray. - Initialize
Now, we calculate the number of subarrays we are looking for by subtracting f(left - 1)
from f(right)
:
- Count of subarrays we are seeking =
f(right)
-f(left - 1)
=4 - 1
=3
.
So, there are 3
subarrays where the maximum element is between 2
and 3
, inclusive. These subarrays are [2]
, [2, 1]
, and [3]
. This method allows us to efficiently calculate the desired count without having to examine each subarray individually.
Solution Implementation
1from typing import List
2
3class Solution:
4 def numSubarrayBoundedMax(self, nums: List[int], left: int, right: int) -> int:
5 # Define a helper function that counts the number of subarrays
6 # with element values less than or equal to 'bound'.
7 def count_subarrays(bound: int) -> int:
8 count = temp_count = 0
9 for value in nums:
10 # Reset temporary count if value is greater than the bound,
11 # or increase it if value is within the subarray criteria.
12 temp_count = 0 if value > bound else temp_count + 1
13 # Add the current temporary count to the total count.
14 count += temp_count
15 return count
16
17 # Calculate the number of subarrays where the maximum element is
18 # between 'left' and 'right' by taking the count of subarrays bounded
19 # by 'right' and subtracting those strictly bounded by 'left - 1'.
20 # This ensures that we only include subarrays where the maximum
21 # value is at least 'left'.
22 return count_subarrays(right) - count_subarrays(left - 1)
23
1class Solution {
2 // Method to count the number of subarrays with elements bounded by left and right
3 public int numSubarrayBoundedMax(int[] nums, int left, int right) {
4 // Calculate the number of subarrays where the maximum element is at most right
5 // and subtract the count of subarrays where the maximum element is less than left
6 return countSubarraysAtMost(nums, right) - countSubarraysAtMost(nums, left - 1);
7 }
8
9 // Helper method to count the number of subarrays with elements at most x
10 private int countSubarraysAtMost(int[] nums, int bound) {
11 int count = 0; // Count of subarrays meeting the condition
12 int currentSubarrayCount = 0; // Temporary count for current valid subarray sequence
13
14 // Iterate through each element of the array
15 for (int num : nums) {
16 // If the current element is not greater than bound, extend the current subarray sequence
17 if (num <= bound) {
18 currentSubarrayCount += 1; // Include this number in the current subarray
19 } else {
20 // If current element is greater than bound, reset the temporary count
21 currentSubarrayCount = 0;
22 }
23
24 // Add the count of valid subarrays up to the current index
25 count += currentSubarrayCount;
26 }
27
28 // Return the total count of valid subarrays
29 return count;
30 }
31}
32
1class Solution {
2public:
3 // Function finds the number of subarrays with elements bounded by left and right.
4 int numSubarrayBoundedMax(vector<int>& nums, int left, int right) {
5 // Lambda function to count subarrays where the max element is less than or equal to a given bound.
6 auto countSubarrays = [&](int bound) {
7 int count = 0; // Number of valid subarrays
8 int currentLength = 0; // Current length of the subarray satisfying the condition
9
10 // Iterate through the elements of the nums vector
11 for (int value : nums) {
12 // If the current value is greater than the bound, reset currentLength to 0
13 // If not, increase currentLength by 1 (extend the valid subarray by the current element)
14 currentLength = value > bound ? 0 : currentLength + 1;
15
16 // Add the length of the current valid subarray to count
17 // This adds all subarray ends with this element and bounded by the value
18 count += currentLength;
19 }
20 return count;
21 };
22
23 // Number of subarrays where the max element is less than or equal to right
24 // and subtracting those where the max element is less than left (to exclude them)
25 return countSubarrays(right) - countSubarrays(left - 1);
26 }
27};
28
1// The type definition for a list of numbers
2type NumberList = number[];
3
4// Function finds the number of subarrays with elements bounded by 'left' and 'right'.
5// Params:
6// nums: The input array of numbers.
7// left: The lower bound for the maximum of the subarrays.
8// right: The upper bound for the maximum of the subarrays.
9// Returns: The count of subarrays meeting the criteria.
10function numSubarrayBoundedMax(nums: NumberList, left: number, right: number): number {
11 // Lambda function to count subarrays where the max element is less than or equal to a given bound.
12 // Params:
13 // bound: The bound to check for in subarrays.
14 // Returns: The count of subarrays where the maximum element is less than or equal to the bound.
15 const countSubarrays = (bound: number): number => {
16 let count: number = 0; // Number of valid subarrays
17 let currentLength: number = 0; // Current length of the subarray satisfying the condition
18
19 // Iterate through the elements of the nums array
20 for (let value of nums) {
21 // If the current value is greater than the bound, reset currentLength to 0
22 // If not, increase currentLength by 1 (extend the valid subarray by the current element)
23 currentLength = value > bound ? 0 : currentLength + 1;
24
25 // Add the length of the current valid subarray to count
26 // This adds all subarray ends with this element and bounded by the value
27 count += currentLength;
28 }
29 return count;
30 };
31
32 // Number of subarrays where the max element is less than or equal to right
33 // and subtracting those where the max element is less than left (to exclude them)
34 return countSubarrays(right) - countSubarrays(left - 1);
35}
36
37// Example usage:
38// const result = numSubarrayBoundedMax([2, 1, 4, 3], 2, 3);
39// console.log(result); // Outputs the count of subarrays meeting the criteria
40
Time and Space Complexity
Time Complexity
The time complexity of the provided code is determined by the function f(x)
which is called twice within the numSubarrayBoundedMax
method. This function iterates through each element of the nums
array exactly once. The number of elements in the nums
array is represented as n
.
Given that there are no nested loops and each operation inside the loop takes constant time, the time taken by f(x)
is proportional to the number of elements in the array, which is O(n)
. Since f(x)
is called twice, the total time complexity is O(2n)
, which simplifies to O(n)
.
Space Complexity
The space complexity of the code can be analyzed by looking at the space taken by variables that could scale with the input. The function f(x)
uses two variables cnt
and t
that do not scale with the size of the input array and are thus constant extra space (O(1)
).
Since there are no additional data structures utilized that grow proportionally with the input size, the space complexity remains constant, irrespective of the size of the input array nums
. Therefore, the space complexity of the code is O(1)
.
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
Tech Interview Pattern Two Pointers Introduction If you prefer videos here's a super quick introduction to Two Pointers div class responsive iframe iframe src https www youtube com embed xZ4AfXHQ1VQ title YouTube video player frameborder 0 allow accelerometer autoplay clipboard write encrypted media gyroscope picture in picture allowfullscreen iframe div Two pointers is a common interview
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.