3095.或值至少K的最短子数组I
滑动窗口,https://leetcode.cn/problems/shortest-subarray-with-or-at-least-k-i/
给你一个 非负 整数数组 nums 和一个整数 k 。
如果一个数组中所有元素的按位或运算 OR 的值 至少 为 k ,那么我们称这个数组是 特别的 。
请你返回 nums 中 最短特别非空
子数组
的长度,如果特别子数组不存在,那么返回 -1 。
示例 1:
输入:nums = [1,2,3], k = 2
输出:1
解释:
子数组 [3] 的按位 OR 值为 3 ,所以我们返回 1 。
注意,[2] 也是一个特别子数组。
示例 2:
输入:nums = [2,1,8], k = 10
输出:3
解释:
子数组 [2,1,8] 的按位 OR 值为 11 ,所以我们返回 3 。
示例 3:
输入:nums = [1,2], k = 0
输出:1
解释:
子数组 [1] 的按位 OR 值为 1 ,所以我们返回 1 。
提示:
1 <= nums.length <= 500 <= nums[i] <= 500 <= k < 64
python
class Solution:
def minimumSubarrayLength(self, nums: List[int], k: int) -> int:
n = len(nums)
left = 0
or_value = 0
min_length = float('inf')
for right in range(n):
or_value |= nums[right]
# 收缩窗口,确保按位或值 >= k
while or_value >= k and left <= right:
min_length = min(min_length, right - left + 1)
left += 1
# 重新计算窗口的按位或值
or_value = 0
for i in range(left, right + 1):
or_value |= nums[i]
return min_length if min_length != float('inf') else -1由于给定数组 nums 中的元素大小不超过 10^9,因此最多需要考虑二进制表示的前 30 位。我们需要维护一个长度为 30 的数组 bits,其中 bits[i] 表示滑动窗口中满足二进制表示的从低到高第 i 位的值为 1 的元素个数。
python
class Solution:
def minimumSubarrayLength(self, nums: List[int], k: int) -> int:
n = len(nums)
bits = [0] * 30
res = inf
def calc(bits):
return sum(1 << i for i in range(30) if bits[i] > 0)
left = 0
for right in range(n):
for i in range(30):
bits[i] += (nums[right] >> i) & 1
while left <= right and calc(bits) >= k:
res = min(res, right - left + 1)
for i in range(30):
bits[i] -= (nums[left] >> i) & 1
left += 1
return -1 if res == inf else res复杂度分析
时间复杂度:O(nlogU),其中 n 表示给定数组 nums 的长度,U 表示数组中的最大的元素。由于使用滑动窗口遍历需要的时间为 O(n),每次更新窗口元素时需要实时计算当前子数组按位或的值需要的时间为 O(logU),此时需要的总时间即为 O(nlogU)。
空间复杂度:O(logU)。计算时需要存储当前子数组中每一个二进制位中的统计情况,最多有 logU 位需要记录,因此需要的空间为 logU。