898. 子数组按位或操作
题目描述
给定一个整数数组 arr
,返回所有 arr
的非空子数组的不同按位或的数量。
子数组的按位或是子数组中每个整数的按位或。含有一个整数的子数组的按位或就是该整数。
子数组 是数组内连续的非空元素序列。
示例 1:
输入:arr = [0] 输出:1 解释: 只有一个可能的结果 0 。
示例 2:
输入:arr = [1,1,2] 输出:3 解释: 可能的子数组为 [1],[1],[2],[1, 1],[1, 2],[1, 1, 2]。 产生的结果为 1,1,2,1,3,3 。 有三个唯一值,所以答案是 3 。
示例 3:
输入:arr = [1,2,4] 输出:6 解释: 可能的结果是 1,2,3,4,6,以及 7 。
提示:
1 <= nums.length <= 5 * 104
0 <= nums[i] <= 109
解法
方法一:哈希表
题目求的是子数组按位或操作的结果的数量,如果我们枚举子数组的结束位置 $i$,那么以 $i-1$ 结尾的子数组按位或操作的结果的数量最多不超过 $32$ 个。这是因为,按位或是一个单调递增的操作。
因此,我们用一个哈希表 $ans$ 记录所有子数组按位或操作的结果,用一个哈希表 $s$ 记录以当前元素结尾的子数组按位或操作的结果,初始时 $s$ 只包含一个元素 $0$。
接下来,我们枚举子数组的结束位置 $i$,那么以 $i$ 结尾的子数组按位或操作的结果,是以 $i-1$ 结尾的子数组按位或操作的结果与 $a[i]$ 进行按位或操作的结果的集合,再加上 $a[i]$ 本身。我们用一个哈希表 $t$ 记录以 $i$ 结尾的子数组按位或操作的结果,然后我们更新 $s = t$,并将 $t$ 中的所有元素加入 $ans$。
最终,我们返回哈希表 $ans$ 中元素的数量即可。
时间复杂度 $O(n \times \log M)$,空间复杂度 $O(n \times \log M)$。其中 $n$ 和 $M$ 分别为数组长度和数组中元素的最大值。
1 2 3 4 5 6 7 8 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
|