2023河南萌新联赛第(八)场 南阳理工学院F小前前




思路:

每次查询L - R的 最大连续子区间的或值

但是 我们都知道或的话 只会增加 不会减少

所以 L - R 的或值就是最大的子区间 然后再求与x的或值就ok了

但是 n=1e5 q=1e5

我们不能每次循环都暴力跑一边 那肯定是不行的

这里我们可以把他们拆成2进制 存放到二维数组里

a[][] 第一层为第i个数 第二层为i层及之前所有数在二进制第j位的和

比如 有3个数

1 2 3

拆成2进制为 00 10 11

那么a[1][0]=0 a[1][1]=0

a[2][0]=1 a[2[1]=0

a[3]0]=2 a[3][1] =1

总共就2^60 完全可以暴力拆解 最大60次

我们在求或值的时候 只需要

a[l] - a[r-1] 即可

就可以获得这个区间的二进制数

然后我们就可以根据第j位是否存在二进制数 来选择是否 sum+=pow(2,j)

总代码如下`

另外需要注意的是 我用内置的pow 会错误

所以用到了快速幂

cpp 复制代码
#include <bits/stdc++.h>

using namespace std;
#define py cout << "YES\n";
typedef long long ll;

int a[100009][66];
int b[65];
ll c[100006];
//快速幂
ll quickPow(ll a, ll n){
    ll ans = 1;
    while(n>0){
        if(n&1>0)        //如果n的当前末位为1
            ans *= a;  //ans乘上当前的a
        a *= a;        //a自乘
        n >>= 1;       //n往右移一位,表示除以2
    }
    return ans;
}
int main(){
	int n,q;
	cin>>n>>q;
	for(int i=1;i<=n;i++){
		ll x;
		cin>>x;
		int cnt=0;
		for(int j=0;j<=62;j++) a[i][j]+=a[i-1][j];
		while(x){
			a[i][cnt++]+=x%2;
			x/=2;
		}
	}
	for(int i=0;i<q;i++){
		int l,r; ll x;
		ll sum=0;
		int t=0;
		memset(b,0,sizeof(b));
		cin>>l>>r>>x;
		while(x){
			b[t++]=x%2;
			x/=2;
		}
		for(int j=0;j<=62;j++){
			b[j]+=a[r][j]-a[l-1][j];
			if(b[j]) sum+=quickPow(2,j);
		//	printf("%d,%d\n",j ,b[j]);
		}
		c[i]=sum;
	}
	for(int i=0;i<q;i++) cout<<c[i]<<'\n';
	
}
相关推荐
追随者永远是胜利者2 小时前
(LeetCode-Hot100)20. 有效的括号
java·算法·leetcode·职场和发展·go
掘根2 小时前
【C++STL】平衡二叉树(AVL树)
开发语言·数据结构·c++
瓦特what?3 小时前
快 速 排 序
数据结构·算法·排序算法
niuniudengdeng3 小时前
基于时序上下文编码的端到端无文本依赖语音分词模型
人工智能·数学·算法·概率论
hetao17338373 小时前
2026-02-13~16 hetao1733837 的刷题记录
c++·算法
你的冰西瓜5 小时前
2026春晚魔术揭秘——变魔法为物理
算法
忘梓.5 小时前
解锁动态规划的奥秘:从零到精通的创新思维解析(10)
c++·算法·动态规划·代理模式
foolish..5 小时前
动态规划笔记
笔记·算法·动态规划
消失的dk5 小时前
算法---动态规划
算法·动态规划
羑悻的小杀马特5 小时前
【动态规划篇】欣赏概率论与镜像法融合下,别出心裁探索解答括号序列问题
c++·算法·蓝桥杯·动态规划·镜像·洛谷·空隙法