XOR on segment(线段树区间异或更新)
本题大意:给定n个数字和m个操作,操作共有两种,第一种是求解区间l到r上元素的和,第二种是将区间l到r的元素都异或一个x,作为某个位置的新值。
很容易想到线段树维护区间和,但是我们发现,在区间更新的时候,并没有办法更新,因为我们并不能通过一个区间的和还有我们要异或的值得到这个区间新的异或和,那么我们考虑其他方法。
下面讲述的这个方法,对于值域不大于1 << 20的数据,我们选择用20个数组保存某一个区间k中第 i 位的1的个数,为什么要这么搞呢?
对于第i位,如果区间k的第 i 位原本有s[ i ][ k ]个1,那么对这个区间所有数字都异或一个数字x,我们只需要判断 x 的每一位,如果存在第 i 位为1,那么s[ i ][ k ] = 区间长度 - s[ i ][ k ]。如果第 i 位为0,那么我们忽略这一位。
why ????
0 xor 1 = 1.... 0 xor 0 = 0 也就是说o异或任何值等于任何值。
1 xor 0 = 1.......1 xor 1 = 0 也就是说与1异或之后值变为相反数。所以区间内1的个数就变为了区间中零的个数。
这样我们就可以轻松统计某个区间某一位上有多少个1。query操作相信应该都是明了的。
那么懒惰标记如何处理呢??
当然是用一个数组lazy_lable记录区间k应该往下推的值,如果该值还未下推又来了另一个值需要改变,那么就将这两个值异或起来,多者不限,也就是说只需要一个一维数组来记录每个区间的lazy值就行了。
ok相信大家都懂了,本来讲之前本人还不怎么懂。。。。。结果讲完就懂了。。。舒服
#include <cstdio>
#include <cstring>
#include <algorithm>
#define mid ((l + r) >> 1)
using namespace std; typedef long long ll; const int maxn = + ; ll segment_tree[][maxn << ], lazy_lable[maxn << ]; int n, m, x; void push_up(int k) {
for(int i = ; i < ; i ++) {
segment_tree[i][k] = segment_tree[i][k << ] + segment_tree[i][k << | ];
}
} void push_down(int k, int son) {
if(lazy_lable[k]) {
lazy_lable[k << ] ^= lazy_lable[k];
lazy_lable[k << | ] ^= lazy_lable[k];
for(int i = ; i < ; i ++) {
if((lazy_lable[k] >> i) & ) {
segment_tree[i][k << ] = son - (son >> ) - segment_tree[i][k << ];
segment_tree[i][k << | ] = (son >> ) - segment_tree[i][k << | ];
}
}
lazy_lable[k] = ;
}
} void build(int l, int r, int k) {
if(l == r) {
scanf("%d", &x);
for(int i = ; i < ; i ++) {
if((x >> i) & )
segment_tree[i][k] = ;
}
return;
}
build(l, mid, k << );
build(mid + , r, k << | );
push_up(k);
} void update(int L, int R, int x, int l, int r, int k) {
if(L <= l && R >= r) {
lazy_lable[k] ^= x;
for(int i = ; i < ; i ++) {
if((x >> i) & ) {
segment_tree[i][k] = r - l + - segment_tree[i][k];
}
}
return;
}
push_down(k, r - l + );
if(L <= mid) update(L, R, x, l, mid, k << );
if(R > mid) update(L, R, x, mid + , r, k << | );
push_up(k);
} ll query(int L, int R, int l, int r, int k) {
if(L <= l && R >= r) {
ll cnt = ;
for(int i = ; i < ; i ++) {
cnt += segment_tree[i][k] << i;
}
return cnt;
}
push_down(k, r - l + );
ll ans = ;
if(L <= mid) ans += query(L, R, l, mid, k << );
if(R > mid) ans += query(L, R, mid + , r, k << | );
return ans;
} int main() {
int op, l, r, x;
scanf("%d", &n);
build(, n, );
scanf("%d", &m);
while(m --) {
scanf("%d", &op);
if(op == ) {
scanf("%d %d", &l, &r);
printf("%lld\n", query(l, r, , n, ));
} else {
scanf("%d %d %d", &l, &r, &x);
update(l, r, x, , n, );
}
}
return ;
}
XOR on segment(线段树区间异或更新)的更多相关文章
- 洛谷 P2574 XOR的艺术(线段树 区间异或 区间求和)
To 洛谷.2574 XOR的艺术 题目描述 AKN觉得第一题太水了,不屑于写第一题,所以他又玩起了新的游戏.在游戏中,他发现,这个游戏的伤害计算有一个规律,规律如下 1. 拥有一个伤害串为长度为n的 ...
- codeforces 22E XOR on Segment 线段树
题目链接: http://codeforces.com/problemset/problem/242/E E. XOR on Segment time limit per test 4 seconds ...
- codeforces 242E. XOR on Segment 线段树
题目链接 给n个数, 两种操作, 一种是求区间内的数的和, 一种是将区间内的数异或x. 异或x没有什么思路, 单个异或肯定超时, 区间异或也没有办法做....后来才知道可以按位建线段树, 这样建20棵 ...
- Codeforces Round #149 (Div. 2) E. XOR on Segment (线段树成段更新+二进制)
题目链接:http://codeforces.com/problemset/problem/242/E 给你n个数,m个操作,操作1是查询l到r之间的和,操作2是将l到r之间的每个数xor与x. 这题 ...
- 洛谷 P4513 小白逛公园-区间最大子段和-分治+线段树区间合并(单点更新、区间查询)
P4513 小白逛公园 题目背景 小新经常陪小白去公园玩,也就是所谓的遛狗啦… 题目描述 在小新家附近有一条“公园路”,路的一边从南到北依次排着nn个公园,小白早就看花了眼,自己也不清楚该去哪些公园玩 ...
- HDU 2795 线段树区间最大值,单点更新+二分
Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- hdu 1116 敌兵布阵 线段树 区间求和 单点更新
线段树的基本知识可以先google一下,不是很难理解 线段树功能:update:单点增减 query:区间求和 #include <bits/stdc++.h> #define lson ...
- HDU 1394:Minimum Inversion Number(线段树区间求和单点更新)
http://acm.hdu.edu.cn/showproblem.php?pid=1394 Minimum Inversion Number Problem Description The in ...
- hdu 3308 线段树 区间合并+单点更新+区间查询
LCIS Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submis ...
随机推荐
- Greenplum 日常维护
1. 数据库启动:gpstart 常用可选参数: -a : 直接启动,不提示终端用户输入确认 -m:只启动master 实例,主要在故障处理时使用 2. 数据库停止:gpstop: 常用可选参数:-a ...
- [深度学习] pytorch学习笔记(2)(梯度、梯度下降、凸函数、鞍点、激活函数、Loss函数、交叉熵、Mnist分类实现、GPU)
一.梯度 导数是对某个自变量求导,得到一个标量. 偏微分是在多元函数中对某一个自变量求偏导(将其他自变量看成常数). 梯度指对所有自变量分别求偏导,然后组合成一个向量,所以梯度是向量,有方向和大小. ...
- Java多线程和并发(六),yield函数和中断线程
目录 1.yield函数 2.中断线程 六.yield函数和中断线程 1.yield函数 2.中断线程 (1)已经被抛弃的方法 (2)目前使用的方法
- 3.决策树ID3算法原理
1.决策树的作用 主要用于解决分类问题的一种算法 2.建立决策树的3中常用算法 1).ID3--->信息增益 2).c4.5--> 信息增益率 4).CART Gini系数 3.提出问题: ...
- 【转载】多网卡的7种bond模式原理
多网卡的7种bond模式原理 Linux 多网卡绑定 网卡绑定mode共有七种(0~6) bond0.bond1.bond2.bond3.bond4.bond5.bond6 常用的有三种 mode=0 ...
- Unity3D_(API)射线检测Raycast()
Unity射线检测官方文档: 传送门 一.检测前方是否有游戏物体(射线无限长度) 二.检测前方是否有游戏物体(射线长度为1m) 三.检测前方游戏物体碰撞信息(射线无限长度): 四.指定检测碰撞Tag层 ...
- scala实战学习-快速排序
def qSort(a:List[Int]):List[Int]={ if(a.length < 2) a else qSort(a.filter(a.head > _)) ++ a.fi ...
- 数据聚类算法-K-means算法
深入浅出K-Means算法 摘要: 在数据挖掘中,K-Means算法是一种 cluster analysis 的算法,其主要是来计算数据聚集的算法,主要通过不断地取离种子点最近均值的算法. K-Mea ...
- kafka监控指标项
监控配置 kafka基本分为broker.producer.consumer三个子项,每一项的启动都需要用到 $KAFKA_HOME/bin/kafka-run-class.sh 脚本,在该脚本中 ...
- js对象深拷贝、浅拷贝
浅拷贝1 //浅拷贝1 let obj01 = { name: 'Lily', age: '20', time: ['13', '15'], person: { name: 'Henry', age: ...