3097. Shortest Subarray With OR at Least K II - Detailed Explanation
Problem Statement
Description:
You are given an integer array A and an integer k. Your goal is to find the length of the shortest contiguous subarray of A whose bitwise OR of all elements is at least k. If no such subarray exists, return -1.
Note:
The bitwise OR of a set of numbers is the result of OR-ing all of them together. Importantly, once a bit is set (i.e., becomes 1) in the OR result, it remains set when more numbers are OR-ed.
Example 1
- Input: A = [1, 2, 3, 4], k = 7
- Explanation:
- Consider subarray [3, 4]:
- 3 in binary:
011
- 4 in binary:
100
- 3 | 4 =
111
in binary, which equals 7.
- 3 in binary:
- The subarray [3, 4] has length 2 and is the shortest that meets the condition.
- Consider subarray [3, 4]:
- Output: 2
Example 2
- Input: A = [8, 1, 2], k = 11
- Explanation:
- The subarray [8, 1] gives 8 | 1 = 9, which is less than 11.
- The only subarray that works is [8, 1, 2]:
- 8 | 1 = 9, and 9 | 2 = 11.
- Its length is 3.
- Output: 3
Example 3
- Input: A = [1, 1, 1], k = 2
- Explanation:
- Any subarray will have OR equal to 1 because 1 | 1 is still 1.
- No subarray reaches the required OR of 2.
- Output: -1
Constraints
- 1 ≤ A.length ≤ 10⁵
- 0 ≤ A[i] ≤ 10⁹
- 0 ≤ k ≤ 10⁹
Hints
-
Understanding OR:
Remember that the bitwise OR is cumulative. If a subarray has built up some bits and you extend it, you can only add more 1’s (never remove them). This makes the OR operation “monotonic” when extending a subarray. -
Building Subarrays Efficiently:
A brute-force method would try every possible subarray—but that’s too slow when A is large. Instead, think about maintaining a set (or dictionary) of distinct OR values for subarrays ending at the current index. When you move to the next index, update these OR values by OR-ing with the new element.
Approach 1: Brute Force (Naive)
Explanation
-
Idea:
Check every possible contiguous subarray. For each subarray, calculate the bitwise OR of its elements. If the OR is at least k, record the subarray’s length. -
Steps:
- Use two nested loops: one for the starting index and one for the ending index.
- For each subarray, compute the OR of the elements.
- Keep track of the minimum subarray length meeting the condition.
- Return the minimum length, or -1 if no subarray qualifies.
-
Drawbacks:
This approach has a time complexity of roughly O(n²) in the worst case, which is too slow when n is large.
Approach 2: Optimized DP/Dictionary Method
Explanation
-
Idea:
For each index in the array, maintain a collection (using a dictionary) of all distinct OR values that can be achieved by subarrays ending at that index. Also, track the earliest starting index for each OR value. -
Steps:
- Initialization:
- Start with an empty dictionary (let’s call it
dp
).
- Start with an empty dictionary (let’s call it
- Iterate Over the Array:
- For each element A[i], create a new dictionary (
new_dp
) where you include the subarray that consists of just A[i] (with starting index i). - Then, for every OR value from the previous index stored in
dp
, compute the new OR by doingprev_OR | A[i]
and record the smallest starting index for this OR value innew_dp
.
- For each element A[i], create a new dictionary (
- Check Condition:
- For every OR value in
new_dp
, if it is at least k, update the answer with the length of the subarray (i.e.,i - start + 1
).
- For every OR value in
- Update dp:
- Set
dp = new_dp
and continue.
- Set
- Initialization:
-
Why It Works:
By iteratively extending the subarrays and merging OR values that are identical, you avoid recalculating the OR for many overlapping subarrays. Since every OR operation is “monotonic” (once a bit is set, it stays set), the number of distinct OR values is bounded by the number of bits in the numbers (typically at most 32 for 32-bit integers).
Code Solutions
Python Code
Java Code
Complexity Analysis
Brute Force Approach
- Time Complexity:
O(n²) in the worst case because every possible subarray is examined. - Space Complexity:
O(1) aside from the input storage.
Optimized DP/Dictionary Approach
- Time Complexity:
For each index, you iterate over the distinct OR values stored in the dictionary. The number of distinct OR values is bounded by the number of bits (typically up to 32). Hence, the time complexity is roughly O(n * 32) ≈ O(n). - Space Complexity:
O(32) per iteration (i.e., O(1) with respect to n) for storing the dictionary of OR values.
Step-by-Step Walkthrough and Visual Example
Consider A = [8, 1, 2] and k = 11:
-
Index 0 (A[0] = 8):
- Start with subarray [8].
dp = {8: 0}
.- Check: 8 < 11, so no update.
-
Index 1 (A[1] = 1):
- Start with [1] → new_dp initially: {1: 1}.
- Extend previous subarray:
- From 8 in dp: 8 | 1 = 9, with starting index 0.
- Now,
new_dp = {1: 1, 9: 0}
. - Check: both 1 and 9 are less than 11.
- Update dp = new_dp.
-
Index 2 (A[2] = 2):
- Start with [2] → new_dp initially: {2: 2}.
- Extend from previous dp:
- From 1: 1 | 2 = 3, starting at index 1.
- From 9: 9 | 2 = 11, starting at index 0.
- Now,
new_dp = {2: 2, 3: 1, 11: 0}
. - Check: 11 ≥ 11 → update answer = 2 - 0 + 1 = 3.
- Final answer is 3.
Common Mistakes
-
Forgetting to Merge OR Values:
Not merging subarrays that produce the same OR can lead to redundant computations and longer run times. -
Incorrect Window Technique:
A typical sliding window approach (used in sum problems) does not directly apply to OR because removing the leftmost element from the OR is not straightforward. -
Edge Case Overlook:
Failing to return -1 when no subarray qualifies.
Edge Cases
-
No Valid Subarray:
When every subarray’s OR is less than k (e.g., all elements are the same and too small), the function should return -1. -
Single Element Array:
Check if the only element is already ≥ k. -
Large k Value:
k might be greater than any OR combination obtainable from the array.
Alternative Variations and Related Problems
-
Variations:
- Longest Subarray with OR Below a Threshold: Find the longest subarray with a bitwise OR less than a given value.
- Subarray with AND at Most k: Use a similar technique with the bitwise AND operator (though AND is “monotonic” in the reverse direction).
-
Related Problems for Further Practice:
- Shortest Subarray with Sum at Least K
- Maximum Subarray Problems
- Subarray Problems Involving Bitwise Operations
GET YOUR FREE
Coding Questions Catalog
