Codeforces Round #442 (Div. 2) E Danil and a Part-time Job (dfs序加上一个线段树区间修改查询)
题意:
给出一个具有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序加上一个线段树区间修改查询)的更多相关文章
- 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 ...
- 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 ...
- Codeforces Round #622 (Div. 2) A. Fast Food Restaurant(全排列,DFS)
Codeforces Round #622 (Div. 2) A. Fast Food Restaurant 题意: 你是餐馆老板,虽然只会做三道菜,上菜时还有个怪癖:一位客人至少上一道菜,且一种菜最 ...
- Codeforces Round #442 (Div. 2) Danil and a Part-time Job
http://codeforces.com/contest/877/problem/E 真的菜的不行,自己敲一个模板,到处都是问题.哎 #include <bits/stdc++.h> u ...
- Codeforces Round #442 Div.2 A B C D E
A. Alex and broken contest 题意 判断一个字符串内出现五个给定的子串多少次. Code #include <bits/stdc++.h> char s[110]; ...
- 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 ...
- Codeforces Round #442 (Div. 2)
A. Alex and broken contest time limit per test 2 seconds memory limit per test 256 megabytes input s ...
- 【Codeforces Round #442 (Div. 2) A】Alex and broken contest
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 注意是所有的名字里面,只出现了其中某一个名字一次. [代码] #include <bits/stdc++.h> usin ...
- 【Codeforces Round #442 (Div. 2) D】Olya and Energy Drinks
[链接] 我是链接,点我呀:) [题意] 给一张二维点格图,其中有一些点可以走,一些不可以走,你每次可以走1..k步,问你起点到终点的最短路. [题解] 不能之前访问过那个点就不访问了.->即k ...
随机推荐
- Ruby版快速排序
class Array def quick_sort return self if self.length<=1 k = self[0] head = 0 tail = self.length ...
- three.js的组合与合并,raycaster射线无法获取group
1.组合 创建一个组非常简单,在组中添加子元素的效果是,你可以对组进行移动.缩放和变形,而所有的子对象都会受到影响.使用组的时候,你依然可以引用.修改每一个单独的几何体.但是,使用raycaster射 ...
- Go 1.11 Module 介绍
title: "Go 1.11 Module" date: 2018-10-26T23:50:56+08:00 draft: false --- Go 1.11 Module 介绍 ...
- 无外接键盘安装 raspberry pi 3B+ 安装系统
从官网介绍看,当前raspbian和以前大家的记录略有不同,老的博客资料基本都是介绍下载raspbian,但现在raspbian已经不再维护镜像, raspbian系统开始由官方 pi foundat ...
- VMware 12 安装 Windows server 2008 系统
一.准备工作 安装了VMware 12 的PC 准备windows server 2008 的ISO操作系统文件 官网地址:https://www.microsoft.com/en-us/downl ...
- 数组Array的API1
数组的方法arr.includes()arr.every(fn(val,i))arr.some(fn(val,i))arr.filter(fn(val,i))arr.map(fn(val,i))ar. ...
- ios监听静音键和音量键事件
http://blog.csdn.net/slinloss/article/details/7870559
- abp.net zero 运行报500.21,错误模块AspNetCoreModuleV2
关于这个运行时提示的问题,导致项目无法运行,之前我是遇到过的,也是查了很久最后解决了 但忘记记录了...岁数大了脑袋不好用了...这次依然找了各种方案,有很多都说由于net core 不是最新的,但我 ...
- PHP+Ajax判断是否有敏感词汇
本文讲述如何使用PHP和Ajax创建一个过滤敏感词汇的方法,判断是否有敏感词汇. 敏感词汇数组sensitive.php return array ( 0 => '111111', 1 => ...
- PHP实现大转盘抽奖算法实例
本文主要向大家介绍了PHP语言实现大转盘抽奖算法,通过具体的实例向大家展示,希望对大家学习PHP抽奖有所帮助. 流程:1.拼装奖项数组,2.计算概率,3.返回中奖情况 代码如下:中奖概率 ' v ' ...