[eJOI2019]异或橙子 题解
简要题面
维护一个数据结构,支持单点修改,询问区间所有子区间的异或和的异或和 .
做法
首先,题目要求所有子区间的异或和的异或和,发现每个元素异或两次就变成 ,所以考虑统计每个元素出现的次数
把区间覆盖元素改成由元素枚举区间,若区间为 ,元素为 ,那么能覆盖到它的就有 个区间(枚举左右端点),即它出现了 次
不难发现 有贡献当且仅当 为奇数,即 均为奇数 . 易见 奇偶性相同 .
维护两个树状数组作奇数位和偶数位即可维护(因为 和 奇偶性相同)
比较方便的办法是令没有的位赋为
注意单点修改 等价于 ,其中 是异或 .
代码
using namespace std;
typedef long long ll;
const int N=2e5+500;
int n,m,arr[N];
template<typename T>
struct BIT
{
private:
T s[N];
inline T lowbit(T x){return x&-x;}
public:
inline T query(T x)
{
T ans=0;
while (x){ans^=s[x]; x-=lowbit(x);}
return ans;
}
inline T query(T l,T r){return query(r)^query(l-1);}
inline void change(int x,T now){if (x) while (x<=n){s[x]^=now; x+=lowbit(x);}}
};
BIT<ll> A,B;
// A 偶数
// B 奇数
void change(int idx,int a)
{
if (idx&1) B.change(idx,a^arr[idx]);
else A.change(idx,a^arr[idx]);
}
ll query(int l,int r)
{
if ((l-r)&1) return 0;
if (l&1) return B.query(l,r);
else return A.query(l,r);
}
int main()
{
scanf("%d%d",&n,&m);
for (int i=1,x;i<=n;i++) scanf("%d",&x),change(i,x),arr[i]=x;
int opt,x,y;
while (m--)
{
scanf("%d%d%d",&opt,&x,&y);
if (opt==1) change(x,y),arr[x]=y;
else printf("%lld\n",query(x,y));
}
return 0;
}
以下是博客签名,正文无关
本文来自博客园,作者:yspm,转载请注明原文链接:https://www.cnblogs.com/CDOI-24374/p/14989007.html
版权声明:本作品采用「署名-非商业性使用-相同方式共享 4.0 国际」许可协议(CC BY-NC-SA 4.0)进行许可。看完如果觉得有用请点个赞吧 QwQ
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】