对线段树的区间更新有了初步的了解。。。

A Simple Tree Problem


Time Limit: 3 Seconds      Memory Limit: 65536 KB

Given a rooted tree, each node has a boolean (0 or 1) labeled on it. Initially, all the labels are 0.

We define this kind of operation: given a subtree, negate all its labels.

And we want to query the numbers of 1's of a subtree.

Input

Multiple test cases.

First line, two integer N and M, denoting the numbers of nodes and numbers of operations and queries.(1<=N<=100000, 1<=M<=10000)

Then a line with N-1 integers, denoting the parent of node 2..N. Root is node 1.

Then M lines, each line are in the format "o node" or "q node", denoting we want to operate or query on the subtree with root of a certain node.

Output

For each query, output an integer in a line.

Output a blank line after each test case.

Sample Input

3 2
1 1
o 2
q 1

Sample Output

1
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <algorithm>
#include <math.h>
#include <map>
#include <queue>
#include <sstream>
#include <iostream>
using namespace std;
#define INF 0x3fffffff
#define N 100100 int n,m;
struct node
{
int to,next;
}edge[*N]; struct node1
{
int b,d,mid;
int flag,sum;
}tnode[*N]; int cnt,pre[N];
int savex[N],savey[N];
int id; // 线段树的区间更新初步了解! void add_edge(int u,int v)
{
edge[cnt].to=v;
edge[cnt].next=pre[u];
pre[u]=cnt++;
} void dfs(int s,int path)
{
int tmp;
++id;
tmp=id;
for(int p=pre[s];p!=-;p=edge[p].next)
{
int v=edge[p].to;
if(v!=path)
{
dfs(v,s);
}
}
savex[s]=tmp;
savey[s]=id;
} void built(int b,int d,int s)
{
tnode[s].b=b;
tnode[s].d=d;
tnode[s].mid=(b+d)/;
tnode[s].sum=;
tnode[s].flag=;
if(b==d) return ;
built(b,tnode[s].mid,*s);
built(tnode[s].mid+,d,*s+);
} void insert(int b,int d,int s)
{
if(tnode[s].b==b&&tnode[s].d==d)
{
tnode[s].flag^=;
tnode[s].sum=(d-b+) - tnode[s].sum;
return ;
}
if(tnode[s].flag!=)
{
tnode[s].flag=;
tnode[*s].flag^=;
tnode[*s].sum=(tnode[*s].d-tnode[*s].b+)-tnode[*s].sum; tnode[*s+].flag^=;
tnode[*s+].sum=(tnode[*s+].d-tnode[*s+].b+)-tnode[*s+].sum;
}
if(tnode[s].mid>=d)
insert(b,d,*s);
else if(tnode[s].mid+<=b)
insert(b,d,*s+);
else
{
insert(b,tnode[s].mid,*s);
insert(tnode[s].mid+,d,*s+);
}
tnode[s].sum=tnode[*s].sum+tnode[*s+].sum;//用up,和down就更好理解
} //这样就不需要放回了,更简单了... int find(int b,int d,int s)
{
if(tnode[s].b==b&&tnode[s].d==d)
{
return tnode[s].sum;
}
if(tnode[s].flag!=)
{
tnode[s].flag=;
tnode[*s].flag^=;
tnode[*s].sum=(tnode[*s].d-tnode[*s].b+)-tnode[*s].sum; tnode[*s+].flag^=;
tnode[*s+].sum=(tnode[*s+].d-tnode[*s+].b+)-tnode[*s+].sum;
}
if(tnode[s].mid>=d)
return find(b,d,s*);
else if(tnode[s].mid+<=b)
return find(b,d,*s+);
else return find(b,tnode[s].mid,s*)+find(tnode[s].mid+,d,*s+);
} int main()
{
//freopen("//home//chen//Desktop//ACM//in.text","r",stdin);
//freopen("//home//chen//Desktop//ACM//out.text","w",stdout);
while(scanf("%d%d",&n,&m)!=EOF)
{
cnt=;
id=;
memset(pre,-,sizeof(pre));
for(int i=;i<=n;i++)
{
int tmp;
scanf("%d",&tmp);
add_edge(i,tmp);
add_edge(tmp,i);
}
dfs(,);
///////////////
built(,n,);
for(int i=;i<m;i++)
{
char sel;
int tmp;
cin>>sel;
scanf("%d",&tmp);
if(sel=='o')
{
insert(savex[tmp],savey[tmp],);
}
else
{
printf("%d\n",find(savex[tmp],savey[tmp],));
}
}
printf("\n");
}
return ;
}

zoj3686(线段树的区间更新)的更多相关文章

  1. hdu 1556:Color the ball(线段树,区间更新,经典题)

    Color the ball Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  2. hdu 1698:Just a Hook(线段树,区间更新)

    Just a Hook Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  3. UVA 12436-Rip Van Winkle's Code(线段树的区间更新)

    题意: long long data[250001]; void A( int st, int nd ) { for( int i = st; i \le nd; i++ ) data[i] = da ...

  4. hdu1698线段树的区间更新区间查询

    Just a Hook Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tota ...

  5. HDU 1556 Color the ball(线段树:区间更新)

    http://acm.hdu.edu.cn/showproblem.php?pid=1556 题意: N个气球,每次[a,b]之间的气球涂一次色,统计每个气球涂色的次数. 思路: 这道题目用树状数组和 ...

  6. Color the ball (线段树的区间更新问题)

    N个气球排成一排,从左到右依次编号为1,2,3....N.每次给定2个整数a b(a <= b),lele便为骑上他的“小飞鸽"牌电动车从气球a开始到气球b依次给每个气球涂一次颜色.但 ...

  7. ZOJ 2301 Color the Ball 线段树(区间更新+离散化)

    Color the Ball Time Limit: 2 Seconds      Memory Limit: 65536 KB There are infinite balls in a line ...

  8. 线段树离散化+区间更新——cf1179C好题

    绝对是很好的题 把问题转化成当第i个询问的答案是数值x时是否可行 要判断值x是否可行,只要再将问题转化成a数组里>=x的值数量是否严格大于b数组里的>=x的值 那么线段树叶子结点维护对于值 ...

  9. hdu 3397 Sequence operation(线段树:区间更新)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3397 题意:给你一个长度为n的0,1序列,支持下列五种操作, 操作0(0 a b):将a到b这个区间的 ...

随机推荐

  1. xaf 自定义登陆页

    web    Model.xafml   view      AuthenticationStandardLogonParameters_DetailView https://documentatio ...

  2. Javascript中的对象和原型(一)(转载)

    面向对象的语言(如Java)中有类的概念,而通过类可以创建任意多个具有相同属性和方法的对象.但是,JavaScript 没有类的概念,因此它的对象也与基于类的语言中的对象有所不同. 要了解面向对象,首 ...

  3. Nginx+Windows负载均衡(转载)

    一.下载Nginxhttp://nginx.org/download/nginx-1.0.8.zip解压到C:\nginx目录下二.在两台服务器上分别建一个网站:S1:192.168.16.35:80 ...

  4. django 官方文档下载

    今天发现自己的django文档有点过时了,于是想去下载一份新,找了老半天才找到pdf的下载地址:于是把它记录下来,希望以后可以直接用. 一.进行django官方网站: 二.文档的pdf文件就在docu ...

  5. 逆向project第005篇:跨越CM4验证机制的鸿沟(下)

    一.前言 本文是逆向分析CM4系列的最后一篇,我会将该游戏的序列号验证机制分析完成,进而编写出注冊码生成器. 二.分析第二个验证循环 延续上一篇文章的内容,来到例如以下代码处: 图1 上述代码并没有特 ...

  6. Eclipse中导入JDK类库的源代码以及添加指定的API

    一.在Eclipse中导入JDK类库的源代码 操作步骤: 打开eclipse->“window”-> “Preferences” -> “Java” -> “Installed ...

  7. Python 切片那些事

  8. Makefile 8——使用依赖关系文件

    Makefile中存在一个include指令,它的作用如同C语言中的#include预处理指令.在Makefile中,可以通过include指令将自动生成的依赖关系文件包含进来,从而使得依赖关系文件中 ...

  9. [转]Vim插件管理工具Vundle

    原文:http://www.linuxzen.com/vimpei-zhi-xi-lie-cha-jian-guan-li.html 当转载成为一种习惯.. 最近对Vim进行了一番较大的配置变动,所以 ...

  10. oozie中时间EL表达式

    EL表达式: 常量表示形式 含义说明 ${coord:minutes(int n)} 返回日期时间:从一开始,周期执行n分钟 ${coord:hours(int n)} 返回日期时间:从一开始,周期执 ...