input: arr[] = {1, 3, 1, 2, 3, 4}, q[] = {{0, 1}, {3, 5}}output:371 or 3 = 32 or 3 or 4 = 7input: arr[] = {1, 2, 3, 4, 5}, q[] = {{0, 4}, {1, 3}}output:77
在给定的问题中,我们将使用强力方法来解决它,然后检查它是否可以适用于更高的约束。如果没有,那么我们将优化我们的方法以适应更高的约束。
暴力方法在这种方法中,我们只需遍历每个范围并计算按位或该范围内的所有数字并打印我们的答案。
示例#include <bits/stdc++.h>using namespace std;int main() { int arr[] = { 7, 5, 3, 5, 2, 3 }; int n = sizeof(arr) / sizeof(int); // size of our array int queries[][2] = { { 1, 3 }, { 4, 5 } }; // given queries int q = sizeof(queries) / sizeof(queries[0]); // number of queries for(int i = 0; i < q; i++) { // traversing through all the queries long ans = 0; for(int j = queries[i][0]; j <= queries[i][1]; j++) // traversing through the range ans |= arr[j]; // calculating the answer cout << ans << "\n"; } return 0;}
输出73
这种方法的时间复杂度为 o(n*q),其中 n 是数组的大小,q 是现在的查询数量,如您所见,这种复杂性不适用于更高的约束,所以现在我们将优化我们的方法,使其也适用于更高的约束。
高效方法在这种方法中,我们将计算前缀位数,然后检查是否数字有一个特定的位集。如果是,那么我们将这一点放入答案中;否则,我们保留这一点。
示例#include <bits/stdc++.h>using namespace std;#define bitt 32#define max (int)10e5int prefixbits[bitt][max];void bitcount(int *arr, int n) { // making prefix counts for (int j = 31; j >= 0; j--) { prefixbits[j][0] = ((arr[0] >> j) & 1); for (int i = 1; i < n; i++) { prefixbits[j][i] = arr[i] & (1ll << j); prefixbits[j][i] += prefixbits[j][i - 1]; } } return;}int check(int l, int r) { // calculating the answer long ans = 0; // to avoid overflow we are taking ans as long for (int i = 0; i < 32; i++) { int x; if (l == 0) x = prefixbits[i][r]; else x = prefixbits[i][r] - prefixbits[i][l - 1]; if (x != 0) ans = (ans | (1ll << i)); } return ans;}int main() { int arr[] = {7, 5, 3, 5, 2, 3}; int n = sizeof(arr) / sizeof(int); // size of our array bitcount(arr, n); int queries[][2] = {{1, 3}, {4, 5}}; // given queries int q = sizeof(queries) / sizeof(queries[0]); // number of queries for (int i = 0; i < q; i++) { cout << check(queries[i][0], queries[i][1]) << "\n"; } return 0;}
输出73
此方法的时间复杂度为 o(n),其中 n 是数组的大小,因此此方法可以适用于更高的约束。
说明上面的代码在这种方法中,我们计算前缀位数并存储它。现在我们计算一个查询,我们遍历该前缀计数并删除 l-1 的位计数,这样我们就有 [l, r] 范围内数字的位计数,因为我们知道如果在任何数字中设置了一个位因此,如果您将其与任何其他数字进行按位或,则该位将保持设置状态,因此使用按位或的此属性,我们检查位计数是否不为零,这意味着范围内存在具有设置位的数字,因此我们设置该位答案并继续循环,最后打印答案。
结论本文解决了计算索引范围 [l, r] 中按位或的查询的问题给定的数组。我们还学习了解决这个问题的c++程序以及解决这个问题的完整方法(正常且高效)。我们可以用其他语言比如c、java、python等语言来编写同样的程序。我们希望这篇文章对您有所帮助。
以上就是使用c++查询给定数组在索引范围内的按位或操作的详细内容。