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:

  1. Initialize cnt and t to 0. cnt will hold the total count of subarrays that do not exceed the maximum allowed element x, and t will count such subarrays ending at the current index in nums.
  2. Iterate over each value v in the nums array.
  3. If v is greater than x, then any subarray including v cannot be part of our solution, so we reset t to 0. This essentially says "you can't start or continue a valid subarray from this point."
  4. If v is less than or equal to x, then we can extend every subarray counted by t with this new value v plus one extra: the subarray that only includes v itself. Thus, we increment t by 1.
  5. For every iteration, we add t to cnt 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 us f(right).
  • Then, we count the subarrays that are strictly smaller than left, which is f(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 Evaluator

Example 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 is f(3):

    1. Initialize cnt = 0 and t = 0.
    2. For v = 2 (first value), t is incremented to 1 (t += 1), so cnt = 1.
    3. For v = 1 (second value), t is incremented to 2 (t += 1), so cnt = 3.
    4. For v = 4 (third value), t is reset to 0, as 4 is greater than 3.
    5. For v = 3 (fourth value), t is incremented to 1 (t += 1), so cnt = 4.

    f(3) gives us a count of 4 subarrays.

  • Using f(left - 1), which is f(1):

    1. Initialize cnt = 0 and t = 0.
    2. For v = 2 (first value), t is reset to 0, since 2 is greater than 1.
    3. For v = 1 (second value), t is incremented to 1 (t += 1), so cnt = 1.
    4. For v = 4 (third value), t is reset to 0, as it's greater than 1.
    5. For v = 3 (fourth value), t is reset again to 0, since 3 is also greater than 1.

    f(1) gives us a count of 1 subarray.

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.


Discover Your Strengths and Weaknesses: Take Our 2-Minute Quiz to Tailor Your Study Plan:
Question 1 out of 10

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

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.

Coding Interview Strategies

Dive into our free, detailed pattern charts and company guides to understand what each company focuses on.

See Patterns
โ†
โ†‘๐Ÿช„