「 Luogu P2574 」 XOR的艺术——线段树
# 解题思路
这题不难,但是原谅我一开始的傻逼想法,一会儿再给大家透露透露。
先说怎么做这题。
显然对于 $0$ 和 $1$ 来说,异或无非也就只有两种变化
- 异或了奇数次,$0$ 就会变成 $1$,$1$ 就会变成 $0$。
- 异或了偶数次,$0$ 和 $1$ 都不变。
那只需要在下传标记的时候下传修改了几次就可以。
好,下面说说我那傻逼的操作。我在下传标记的时候没有给子节点的 sum 进行异或,而是只下传了标记,并且在回溯的时候没有更新父节点的 sum 值。
然后我成功的没过样例,然鹅这并不傻逼,改过来就行了吗,对吧,但是,傻逼的来了,我改的时候没有按照上面的来改,而是在进行更新时,沿途将节点打一个flag标记,假装这一段区间需要更改。
哈哈哈哈哈,然后我成功的获得了 T 四个点的好成绩。
什么鬼,我居然忘记更新父节点,果然还是我太菜了。
我决定还是把这个代码放上吧。
# 附上代码
放上我的傻逼错误代码
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int n, m;
const int maxn = 2e5+;
struct node {int l, r, sum, t, tag, flag;}tree[maxn << ];
struct Tree {
#define Lson (k << 1)
#define Rson ((k << 1) | 1)
template <typename T> inline void read(T &x) {
x = ; T f = ; char c = getchar();
while (c < '' || c > '') {if(c == '-') f = -; c = getchar();}
while (c <= '' && c >= '') {x = x* + c-''; c = getchar();}
x *= f;
}
void build(int k, int ll, int rr) {
tree[k].l = ll, tree[k].r = rr;
tree[k].flag = tree[k].t = ;
if(tree[k].l == tree[k].r) {
scanf("%1d", &tree[k].sum);
return ;
}
int mid = (tree[k].l + tree[k].r) >> ;
build(Lson, tree[k].l, mid);
build(Rson, mid+, tree[k].r);
tree[k].sum = tree[Lson].sum + tree[Rson].sum;
}
void push_down(int k) {
tree[Lson].t += tree[k].tag;
tree[Rson].t += tree[k].tag;
tree[Lson].tag += tree[k].tag;
tree[Rson].tag += tree[k].tag;
tree[k].tag = ;
}
void update(int k, int L, int R) {
if(tree[k].l >= L && tree[k].r <= R) {
tree[k].t ++;
tree[k].tag ++;
return;
}
tree[k].flag = ;
if(tree[k].tag) push_down(k);
int mid = (tree[k].l + tree[k].r) >> ;
if(L <= mid) update(Lson, L, R);
if(R > mid) update(Rson, L, R);
}
int query(int k, int L, int R) {
int ans = ;
if(tree[k].l >= L && tree[k].r <= R && tree[k].flag == ) {
if(tree[k].t % == ) return (tree[k].r-tree[k].l+)-tree[k].sum;
else return tree[k].sum;
}
if(tree[k].tag) push_down(k);
int mid = (tree[k].l + tree[k].r) >> ;
if(L <= mid) ans += query(Lson, L, R);
if(R > mid) ans += query(Rson, L, R);
return ans;
}
Tree () {
read(n), read(m);
build(, , n);
int opt, l, r;
for(int i=; i<=m; i++) {
read(opt), read(l), read(r);
if(opt == ) update(, l, r);
else printf("%d\n", query(, l, r));
}
}
}T;
int main() {return ;}
好,我们再来看看正确的代码
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int n, m;
const int maxn = 2e5+;
struct node {int l, r, sum, t, tag, flag;}tree[maxn << ];
struct Tree {
#define Lson (k << 1)
#define Rson ((k << 1) | 1)
template <typename T> inline void read(T &x) {
x = ; T f = ; char c = getchar();
while (c < '' || c > '') {if(c == '-') f = -; c = getchar();}
while (c <= '' && c >= '') {x = x* + c-''; c = getchar();}
x *= f;
}
void build(int k, int ll, int rr) {
tree[k].l = ll, tree[k].r = rr;
tree[k].flag = tree[k].t = ;
if(tree[k].l == tree[k].r) {
scanf("%1d", &tree[k].sum);
return ;
}
int mid = (tree[k].l + tree[k].r) >> ;
build(Lson, tree[k].l, mid);
build(Rson, mid+, tree[k].r);
tree[k].sum = tree[Lson].sum + tree[Rson].sum;
}
void push_down(int k) {
if(tree[k].tag % == )
tree[Lson].sum = (tree[Lson].r-tree[Lson].l+)-tree[Lson].sum,
tree[Rson].sum = (tree[Rson].r-tree[Rson].l+)-tree[Rson].sum;
tree[Lson].tag += tree[k].tag;
tree[Rson].tag += tree[k].tag;
tree[k].tag = ;
}
void update(int k, int L, int R) {
if(tree[k].l >= L && tree[k].r <= R) {
tree[k].tag ++;
tree[k].sum = (tree[k].r-tree[k].l+)-tree[k].sum;
return;
}
if(tree[k].tag) push_down(k);
int mid = (tree[k].l + tree[k].r) >> ;
if(L <= mid) update(Lson, L, R);
if(R > mid) update(Rson, L, R);
tree[k].sum = tree[Lson].sum + tree[Rson].sum;
}
int query(int k, int L, int R) {
int ans = ;
if(tree[k].l >= L && tree[k].r <= R)
return tree[k].sum;
if(tree[k].tag) push_down(k);
int mid = (tree[k].l + tree[k].r) >> ;
if(L <= mid) ans += query(Lson, L, R);
if(R > mid) ans += query(Rson, L, R);
return ans;
}
Tree () {
read(n), read(m);
build(, , n);
int opt, l, r;
for(int i=; i<=m; i++) {
read(opt), read(l), read(r);
if(opt == ) update(, l, r);
else printf("%d\n", query(, l, r));
}
}
}T;
int main() {return ;}
「 Luogu P2574 」 XOR的艺术——线段树的更多相关文章
- luogu P2574 XOR的艺术 (线段树)
luogu P2574 XOR的艺术 (线段树) 算是比较简单的线段树. 当区间修改时.\(1 xor 1 = 0,0 xor 1 = 1\)所以就是区间元素个数减去以前的\(1\)的个数就是现在\( ...
- 洛谷 P2574 XOR的艺术(线段树 区间异或 区间求和)
To 洛谷.2574 XOR的艺术 题目描述 AKN觉得第一题太水了,不屑于写第一题,所以他又玩起了新的游戏.在游戏中,他发现,这个游戏的伤害计算有一个规律,规律如下 1. 拥有一个伤害串为长度为n的 ...
- LOJ 2991 「THUSC 2016」补退选——trie+线段树合并或vector
题目:https://loj.ac/problem/2291 想了线段树合并的做法.就是用线段树维护 trie 的每个点在各种时间的操作. 然后线段树合并一番,线段树维护前缀最大值,就是维护最大子段和 ...
- 【LOJ6620】「THUPC 2019」不等式 / inequality(线段树)
点此看题面 大致题意: 给你两个长度为\(n\)的数组\(a_i\)和\(b_i\),定义\(f_k(x)=\sum_{i=1}^k|a_ix+b_i|\),对于\(k=1\sim n\)的每个\(f ...
- 【LOJ】#3109. 「TJOI2019」甲苯先生的线段树
LOJ#3109. 「TJOI2019」甲苯先生的线段树 发现如果枚举路径两边的长度的话,如果根节点的值是$x$,左边走了$l$,右边走了$r$ 肯定答案会是$(2^{l + 1} + 2^{r + ...
- 【洛谷P2574】XOR的艺术
XOR的艺术 题目链接 用线段树维护sum, 修改时 tag[p]^=1; sum=r-l+1-sum; 详见代码 #include<iostream> #include<cstdi ...
- 「Luogu 3792」由乃与大母神原型和偶像崇拜
更好的阅读体验 Portal Portal1: Luogu Description 给你一个序列\(a\) 每次两个操作: 修改\(x\)位置的值为\(y\): 查询区间\([l, r]\)是否可以重 ...
- loj#2312. 「HAOI2017」八纵八横(线性基 线段树分治)
题意 题目链接 Sol 线性基+线段树分治板子题.. 调起来有点自闭.. #include<bits/stdc++.h> #define fi first #define se secon ...
- 「POI2011 R2 Day2」Tree Rotations【线段树合并】
题目链接 [BZOJ] [洛谷] [LOJ] 题解 由于是前序遍历,那么讨论一棵树上的逆序对的情况. 两个节点都在左子树上 两个节点都在右子树上 两个节点分别在不同的子树上. 前两种情况其实也可以归结 ...
随机推荐
- Android Handler消息机制源码解析
好记性不如烂笔头,今天来分析一下Handler的源码实现 Handler机制是Android系统的基础,是多线程之间切换的基础.下面我们分析一下Handler的源码实现. Handler消息机制有4个 ...
- 51nod1014【暴力】
直接暴力1e6就好了 #include <bits/stdc++.h> using namespace std; typedef long long LL; int main() { LL ...
- Hello world——程序员的第一篇代码
Hello world 是和A+B问题并驾齐驱的一道题,也是当世的经典题之一. 题目: 输出“Hello world” 样例输入 样例输出 Hello world Hello world ...
- postman接口测试系列:环境配置
最近忙着项目接口测试,经过不同工具的对比,发现postman使用起来挺顺手的,所以马上决定使用这个工具进行接口测试工作.刚开始的时候,了解了下接口测试的相关信息,直接着手编写接口测试的测试用例信息 ...
- April Fools Contest 2017 F
Description You are developing a new feature for the website which sells airline tickets: being able ...
- window上安装MySQL
一.安装MySQL 1.1 下载解压缩版的安装包,解压,然后配置环境变量 PATH=.......;D:\Program Files (x86)\mysql-5.5.27-win32\bin (注意是 ...
- C#基础学习5
数组集合
- SonarQube+Svn+Jenkins环境搭建----问题总结
1.配置SVN后提示unable to access to repository,原因是使用的账户没有访问svn的权限,创建新的用户即可.注意新的用户,用户名,密码要跟svn上的权限一致. 创 ...
- idea web项目热部署
之前用idea写web项目的时候,一直都是改一点东西就要重启一下,很烦.今天终于忍受不了百度了一下idea怎么热部署web项目. 在此记录下. 第一步 编辑tomcat配置 第二步 选择打包的项目,并 ...
- 基于Java实现的冒泡排序算法
冒泡排序是一种简单基础的排序算法,相信在大学课堂里老师已经讲过了,现在我基于Java来实现一遍. 简述 冒泡排序正如其关键词一样,杂乱的气泡经过浮动,最后大的气泡飘到了上面而小的气泡在下面,无序的元素 ...