「 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+3;
struct node {int l, r, sum, t, tag, flag;}tree[maxn << 2];
struct Tree {
    #define Lson (k << 1)
    #define Rson ((k << 1) | 1)
    template <typename T> inline void read(T &x) {
        x = 0; T f = 1; char c = getchar();
        while (c < ‘0‘ || c > ‘9‘) {if(c == ‘-‘) f = -1; c = getchar();}
        while (c <= ‘9‘ && c >= ‘0‘) {x = x*10 + c-‘0‘; 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 = 0;
        if(tree[k].l == tree[k].r) {
            scanf("%1d", &tree[k].sum);
            return ;
        }
        int mid = (tree[k].l + tree[k].r) >> 1;
        build(Lson, tree[k].l, mid);
        build(Rson, mid+1, 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 = 0;
    }
    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 = 1;
        if(tree[k].tag) push_down(k);
        int mid = (tree[k].l + tree[k].r) >> 1;
        if(L <= mid) update(Lson, L, R);
        if(R > mid) update(Rson, L, R);
    }
    int query(int k, int L, int R) {
        int ans = 0;
        if(tree[k].l >= L && tree[k].r <= R && tree[k].flag == 0) {
            if(tree[k].t % 2 == 1) return (tree[k].r-tree[k].l+1)-tree[k].sum;
            else return tree[k].sum;
        }
        if(tree[k].tag) push_down(k);
        int mid = (tree[k].l + tree[k].r) >> 1;
        if(L <= mid) ans += query(Lson, L, R);
        if(R > mid) ans += query(Rson, L, R);
        return ans;
    }
    Tree () {
        read(n), read(m);
        build(1, 1, n);
        int opt, l, r;
        for(int i=1; i<=m; i++) {
            read(opt), read(l), read(r);
            if(opt == 0) update(1, l, r);
            else printf("%d\n", query(1, l, r));
        }
    }
}T;
int main() {return 0;}

好,我们再来看看正确的代码

#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int n, m;
const int maxn = 2e5+3;
struct node {int l, r, sum, t, tag, flag;}tree[maxn << 2];
struct Tree {
    #define Lson (k << 1)
    #define Rson ((k << 1) | 1)
    template <typename T> inline void read(T &x) {
        x = 0; T f = 1; char c = getchar();
        while (c < ‘0‘ || c > ‘9‘) {if(c == ‘-‘) f = -1; c = getchar();}
        while (c <= ‘9‘ && c >= ‘0‘) {x = x*10 + c-‘0‘; 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 = 0;
        if(tree[k].l == tree[k].r) {
            scanf("%1d", &tree[k].sum);
            return ;
        }
        int mid = (tree[k].l + tree[k].r) >> 1;
        build(Lson, tree[k].l, mid);
        build(Rson, mid+1, tree[k].r);
        tree[k].sum = tree[Lson].sum + tree[Rson].sum;
    }
    void push_down(int k) {
        if(tree[k].tag % 2 == 1)
            tree[Lson].sum = (tree[Lson].r-tree[Lson].l+1)-tree[Lson].sum,
            tree[Rson].sum = (tree[Rson].r-tree[Rson].l+1)-tree[Rson].sum;
        tree[Lson].tag += tree[k].tag;
        tree[Rson].tag += tree[k].tag;
        tree[k].tag = 0;
    }
    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+1)-tree[k].sum;
            return;
        }
        if(tree[k].tag) push_down(k);
        int mid = (tree[k].l + tree[k].r) >> 1;
        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 = 0;
        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) >> 1;
        if(L <= mid) ans += query(Lson, L, R);
        if(R > mid) ans += query(Rson, L, R);
        return ans;
    }
    Tree () {
        read(n), read(m);
        build(1, 1, n);
        int opt, l, r;
        for(int i=1; i<=m; i++) {
            read(opt), read(l), read(r);
            if(opt == 0) update(1, l, r);
            else printf("%d\n", query(1, l, r));
        }
    }
}T;
int main() {return 0;}

原文地址:https://www.cnblogs.com/bljfy/p/9790383.html

时间: 2024-11-06 09:54:31

「 Luogu P2574 」 XOR的艺术——线段树的相关文章

「Luogu P3178」[HAOI2015]树上操作

有一棵点数为 \(N\) 的树,以点 \(1\) 为根,且树点有边权.然后有 \(M\) 个操作,分为三种: 操作 1 :把某个节点 \(x\) 的点权增加 \(a\) . 操作 2 :把某个节点 \(x\) 为根的子树中所有点的点权都增加 \(a\) . 操作 3 :询问某个节点 \(x\) 到根的路径中所有点的点权和. Luogu 分析 我们把树上问题利用 \(dfs\) 序转化成序列问题然后直接上线段树解决即可. 考虑将线段树的每个叶子结点设为在原树上的点到根的点权和.对于单点修改,当前结

loj2537 「PKUWC2018」Minimax 【概率 + 线段树合并】

题目链接 loj2537 题解 观察题目的式子似乎没有什么意义,我们考虑计算出每一种权值的概率 先离散化一下权值 显然可以设一个\(dp\),设\(f[i][j]\)表示\(i\)节点权值为\(j\)的概率 如果\(i\)是叶节点显然 如果\(i\)只有一个儿子直接继承即可 如果\(i\)有两个儿子,对于儿子\(x\),设另一个儿子为\(y\) 则有 \[f[i][j] += f[x][j](1 - p_i)\sum\limits_{k > j}f[r][k] + f[x][j]p_i\sum\

loj#2312. 「HAOI2017」八纵八横(线性基 线段树分治)

题意 题目链接 Sol 线性基+线段树分治板子题.. 调起来有点自闭.. #include<bits/stdc++.h> #define fi first #define se second #define pb push_back #define bit bitset<B + 1> using namespace std; const int MAXN = 501, B = 1001, SS = 4001; inline int read() { char c = getchar

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. 这题是线段树成段更新,但是不能直接更新,不然只能一个数一个数更新.这样只能把每个数存到一个数组中,长度大概是20吧,然后模拟二进制的位操作.仔细一点就行了. 1 #include <iostream> 2 #include <cstdio> 3 #include <cmath>

codeforces 242E. XOR on Segment 线段树

题目链接 给n个数, 两种操作, 一种是求区间内的数的和, 一种是将区间内的数异或x. 异或x没有什么思路, 单个异或肯定超时, 区间异或也没有办法做....后来才知道可以按位建线段树, 这样建20棵线段树就可以. 每一次异或, 对于给定的x, 如果x的第i位是1, 那么就将第i棵线段树在给定的区间内0,1翻转, 这是很基础的操作. 对于区间求和操作, 我们可以求出给定的区间, 从高位到低位, 每一位依次有多少个1, 然后就可以直接求出来, 感觉不好表达....具体看代码. 1 #include

[luogu P3801] 红色的幻想乡 [线段树][树状数组]

题目背景 蕾米莉亚的红雾异变失败后,很不甘心. 题目描述 经过上次失败后,蕾米莉亚决定再次发动红雾异变,但为了防止被灵梦退治,她决定将红雾以奇怪的阵势释放. 我们将幻想乡看做是一个n*m的方格地区,一开始没有任何一个地区被红雾遮盖.蕾米莉亚每次站在某一个地区上,向东南西北四个方向各发出一条无限长的红雾,可以影响到整行/整列,但不会影响到她所站的那个地区.如果两阵红雾碰撞,则会因为密度过大而沉降消失.灵梦察觉到了这次异变,决定去解决它.但在解决之前,灵梦想要了解一片范围红雾的密度.可以简述为两种操

[luogu P3797] 妖梦斩木棒 [线段树]

题目背景 妖梦是住在白玉楼的半人半灵,拥有使用剑术程度的能力. 题目描述 有一天,妖梦正在练习剑术.地面上摆放了一支非常长的木棒,妖梦把它们切成了等长的n段.现在这个木棒可以看做由三种小段构成,中间的n-2段都是左右都被切断的断头,我们记做’X’,最左边的一段和最右边的一段各有一个圆头,记做’(‘和’)’.幽幽子吃饱后闲来无事,决定戏弄一下妖梦.她拿来了许多这样的三种小段木棒,来替换掉妖梦切下来的n段中的一部分,然后问妖梦一些问题.这些操作可以这样描述: 1 x C 将第x个小段的木棒替换成C型

Luogu P5416 [CTSC2016]时空旅行(线段树分治)

题目 简化题意:你需要维护\(n\)个集合,集合元素为二元组\((x,v)\).集合\(i\)的产生方式是以某个原有集合\(p_i\)为样本,扩展或删除一个元素后得到新集合.有\(q\)次询问,每次给出\(y\)并指定一个集合\(i\),要求从集合\(i\)中找出一个元素,最小化\((x?y)^2+v\). 先拆式子\((x-y)^2+v=x^2-2xy+y^2+v\),令其等于\(k\)即\(x^2+y^2-2xy+v=k\). 移项得\(2yx+k=y^2+x^2+v\),可以看作是\((x

「Luogu 1821」[USACO07FEB]银牛派对Silver Cow Party

更好的阅读体验 Portal Portal1: Luogu Portal2: POJ Description One cow from each of N farms \((1 \le N \le 1000)\) conveniently numbered \(1 \cdots N\) is going to attend the big cow party to be held at farm #X \((1 \le X \le N)\). A total of \(M (1 \le M \l