题意:

给出一个具有N个点的树,现在给出两种操作:

1.get x,表示询问以x作为根的子树中,1的个数。

2.pow x,表示将以x作为根的子树全部翻转(0变1,1变0).

思路:dfs序加上一个线段树区间修改查询。

AC代码:

#include<iostream>
#include<vector>
#include<string.h>
using namespace std;
const int maxn=2e5+5;
int sum[maxn<<2],lazy[maxn<<2],a[maxn],num[maxn],lft[maxn],rght[maxn],tot;
char s[10];
vector<int>map[maxn];

void build(int k,int l,int r){
    if(l==r){
        sum[k]=a[num[l]];
        return ;
    }
    int mid=(l+r)/2;
    build(k*2,l,mid);
    build(k*2+1,mid+1,r);
    sum[k]=sum[k*2]+sum[k*2+1];
}

void pushdown(int k,int l,int r){
    if(lazy[k]){
        int mid=(l+r)/2;
        sum[k*2]=mid-l+1-sum[k*2];
        sum[k*2+1]=r-mid-sum[k*2+1];
        //sum[k]=sum[k*2]+sum[k*2+1];
        lazy[k*2]^=1;
        lazy[k*2+1]^=1;
        lazy[k]=0;
    }
}

void update(int k,int l,int r,int L,int R){
    if(L<=l&&r<=R){
        sum[k]=r-l+1-sum[k];
        lazy[k]^=1;
        return;
    }
    pushdown(k,l,r);
    int mid=(l+r)/2;
    if(mid>=L)update(k*2,l,mid,L,R);
    if(mid<R)update(k*2+1,mid+1,r,L,R);
    sum[k]=sum[k*2]+sum[k*2+1];
    return ;
}

int query(int k,int l,int r,int L,int R){
    if(L<=l&&r<=R)return sum[k];
    int ans=0;
    pushdown(k,l,r);
    int mid=(l+r)/2;
    if(mid>=L)ans+=query(k*2,l,mid,L,R);
    if(mid<R)ans+=query(k*2+1,mid+1,r,L,R);
    //sum[k]=sum[k*2]+sum[k*2+1];
    return ans;
}
//pushdown和query的pushup可以不要,但是在update中pushdown之后需要pushup
 
void dfs(int k){
    lft[k]=++tot;
    for(int i=0;i<map[k].size();i++){
        dfs(map[k][i]);
    }
    rght[k]=tot;
}
//每个点的left序号就是在线段树中的序号
//每个点对应的left[i]-right[i]+1就是每个点的字节点的个数(加上自己)
 
int main(){
    int n,x;
    while(cin>>n){
        for(int i=1;i<=n;i++)map[i].clear();
        for(int i=2;i<=n;i++){
            cin>>x;
            map[x].push_back(i);
        }
        memset(sum,0,sizeof(sum));
        memset(lazy,0,sizeof(lazy));
        tot=0;
        dfs(1);
        /*for(int i=1;i<=n;i++){
            cout<<lft[i]<<' '<<rght[i]<<endl;
        }*/
        for(int i=1;i<=n;i++){
            cin>>a[i];
        }
        for(int i=1;i<=n;i++){
            num[lft[i]]=i;
            //每个点在线段树中的序号和本身不是一样的
            //注意看build函数
        }
        build(1,1,n);
        int q;
        cin>>q;
        for(int i=0;i<q;i++){
            cin>>s>>x;
            if(s[0]=='p'){
                update(1,1,n,lft[x],rght[x]);
            }
            else cout<<query(1,1,n,lft[x],rght[x])<<endl;
        }
    }
    return 0;
}

圆圈中的数字是每个节点的left值,每个点的right值就是每个点的子节点中left最大的那个

该图对应输入

10
1 2 3 4 2 4 1 7 8
1 1 0 1 1 0 0 0 1 1
10
pow 1
get 2
pow 2
pow 8
get 6
pow 6
pow 10
get 6
pow 8
pow 3
输出 3 0 1

(第一次看dfs序,看了好久才明白,还耽误了吴老狗睡觉的时间T^T,还是太菜了)

Codeforces Round #442 (Div. 2) E Danil and a Part-time Job (dfs序加上一个线段树区间修改查询)的更多相关文章

  1. Codeforces Round #169 (Div. 2) E. Little Girl and Problem on Trees dfs序+线段树

    E. Little Girl and Problem on Trees time limit per test 2 seconds memory limit per test 256 megabyte ...

  2. Codeforces Round #442 (Div. 2)A,B,C,D,E(STL,dp,贪心,bfs,dfs序+线段树)

    A. Alex and broken contest time limit per test 2 seconds memory limit per test 256 megabytes input s ...

  3. Codeforces Round #622 (Div. 2) A. Fast Food Restaurant(全排列,DFS)

    Codeforces Round #622 (Div. 2) A. Fast Food Restaurant 题意: 你是餐馆老板,虽然只会做三道菜,上菜时还有个怪癖:一位客人至少上一道菜,且一种菜最 ...

  4. Codeforces Round #442 (Div. 2) Danil and a Part-time Job

    http://codeforces.com/contest/877/problem/E 真的菜的不行,自己敲一个模板,到处都是问题.哎 #include <bits/stdc++.h> u ...

  5. Codeforces Round #442 Div.2 A B C D E

    A. Alex and broken contest 题意 判断一个字符串内出现五个给定的子串多少次. Code #include <bits/stdc++.h> char s[110]; ...

  6. Codeforces Round #877 (Div. 2) E. Danil and a Part-time Job

    E. Danil and a Part-time Job 题目链接:http://codeforces.com/contest/877/problem/E time limit per test2 s ...

  7. Codeforces Round #442 (Div. 2)

    A. Alex and broken contest time limit per test 2 seconds memory limit per test 256 megabytes input s ...

  8. 【Codeforces Round #442 (Div. 2) A】Alex and broken contest

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 注意是所有的名字里面,只出现了其中某一个名字一次. [代码] #include <bits/stdc++.h> usin ...

  9. 【Codeforces Round #442 (Div. 2) D】Olya and Energy Drinks

    [链接] 我是链接,点我呀:) [题意] 给一张二维点格图,其中有一些点可以走,一些不可以走,你每次可以走1..k步,问你起点到终点的最短路. [题解] 不能之前访问过那个点就不访问了.->即k ...

随机推荐

  1. 工控随笔_10_西门子_WinCC的VBS脚本_01_基础入门

    很多人都认为VB语言或者VBS脚本语言是一种很low的语言,从心里看不起VB或者VBS, 但是其实VBS不仅可以做为系统管理员的利器,同样在工控领域VBS语言大有用武之地. 西门子的WinCC提供了两 ...

  2. mac下 chmod 不了文件,可以右键查看文件简介处修改权限

  3. nginx+uwsgi配置

    nginx #user nobody; worker_processes 1; #error_log logs/error.log; #error_log logs/error.log notice; ...

  4. window注册表相关

    参考: https://baike.baidu.com/item/REG_EXPAND_SZ/9102962 一 注册表的相关概念 windows注册表相关api中名字起的比较混乱, 在这放一张从网上 ...

  5. 序列化模块_pickle

    序列化: 把不能够直接存储的数据变成字节流(bytes)保存在文件, 进行持久化存储 反序列化: 任何数据都可以转成字节流(bytes)进行存储: 1. dumps 把任意对象序列化 li = [1, ...

  6. 一台机器部署多个tomcat服务 nginx反向代理多个服务 笔记

    安装tomcat步骤           1. 下载apache-tomcat-8.0.30 ,下载下来的文件为apache-tomcat-8.0.30-windows-x64.zip         ...

  7. 安装Git Bash图文教程

    1.下载Git Bash,下载地址 https://pan.baidu.com/s/1sllsi0d 2.双击Git-2.9.2-64-bit.exe,运行,进行安装:点击“Next” 3.设置安装路 ...

  8. WordPress版微信小程序3.1.5版的新功能

    产品的完善是无止境,每过段时间就会发现产品的新问题,使用的人越多,提的需求也会越多,我听得最多的一句话就是:如果加上某某功能就完美了.其实,完美是不存在的,每个人的视角不一样,完美的定义也是不一样的. ...

  9. Verilog风格

    Verilog HDL编写原则: 可移植性强(多用宏定义:少用嵌入代码中的常数,即Magic Number:使用头文件): 必要的注释,代码易读: 模块间耦合尽可能低: 变量名与宏的命名规则: 变量名 ...

  10. Linux中VIM的使用

    转自:http://www.lupaworld.com/?uid-296380-action-viewspace-itemid-118973 vi/vim 基本使用方法本文介绍了vi (vim)的基本 ...