标签:eJOI2019 return idx int 题解 异或 query change
简要题面
维护一个数据结构,支持单点修改,询问区间所有子区间的异或和的异或和 .
做法
首先,题目要求所有子区间的异或和的异或和,发现每个元素异或两次就变成 \(0\),所以考虑统计每个元素出现的次数
把区间覆盖元素改成由元素枚举区间,若区间为 \([l,r]\),元素为 \(a_i\),那么能覆盖到它的就有 \((l-i+1)(r-i+1)\) 个区间(枚举左右端点),即它出现了 \((l-i+1)(r-i+1)\) 次
不难发现 \(i\) 有贡献当且仅当 \((l-i+1)(r-i+1)\) 为奇数,即 \(l-i+1,r-i+1\) 均为奇数 . 易见 \(l,r,i\) 奇偶性相同 .
维护两个树状数组作奇数位和偶数位即可维护(因为 \(i\) 和 \(l,r\) 奇偶性相同)
比较方便的办法是令没有的位赋为 \(0\)
注意单点修改 \(a_i\gets k\) 等价于 \(a_i\gets a_i\oplus(a_i\oplus k)\),其中 \(\oplus\) 是异或 .
代码
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;
}
标签:eJOI2019,return,idx,int,题解,异或,query,change 来源: https://www.cnblogs.com/CDOI-24374/p/14989007.html
本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享; 2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关; 3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关; 4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除; 5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。