题目链接

描述

There is an apple tree outside of kaka's house. Every autumn, a lot of apples will grow in the tree. Kaka likes apple very much, so he has been carefully nurturing the big apple tree.

The tree has N forks which are connected by branches. Kaka numbers the forks by 1 to N and the root is always numbered by 1. Apples will grow on the forks and two apple won't grow on the same fork. kaka wants to know how many apples are there in a sub-tree, for his study of the produce ability of the apple tree.

The trouble is that a new apple may grow on an empty fork some time and kaka may pick an apple from the tree for his dessert. Can you help kaka?

输入

The first line contains an integer N (N ≤ 100,000) , which is the number of the forks in the tree.

The following N - 1 lines each contain two integers u and v, which means fork u and fork v are connected by a branch.

The next line contains an integer M (M ≤ 100,000).

The following M lines each contain a message which is either

"C x" which means the existence of the apple on fork x has been changed. i.e. if there is an apple on the fork, then Kaka pick it; otherwise a new apple has grown on the empty fork.

or

"Q x" which means an inquiry for the number of apples in the sub-tree above the fork x, including the apple (if exists) on the fork x

Note the tree is full of apples at the beginning

输出

For every inquiry, output the correspond answer per line.

样例输入

3

1 2

1 3

3

Q 1

C 2

Q 1

样例输出

3

2

分析:

有一棵苹果树,刚开始的时候每个节点上都有且仅有一颗苹果,需要进行一系列操作,Q N表示的是要查询N节点下有多少个苹果(包括N节点本身和它下面所有的子节点),C N表示的是如果N节点上有苹果的话,就把这个苹果拿走,否则就往这个节点上放一个苹果。

在查询的时候,要求出这个节点下的所有的苹果树,相当于求的是一个区间和,如果给这棵树的每个节点都给合理的编号,那么就可以按照树状数组来求区间和了。

Step 1:

如下图,可以看到,由于普通的树并不具有区间的性质,所以在考虑使用线段树作为解题思路时,需要对给给定的数据进行转化,首先对这棵树进行一次dfs遍历,记录dfs序下每个点访问起始时间与结束时间,记录起始时间是前序遍历,结束时间是后序遍历,同时对这课树进行重标号。

Step 2:

     如下图,DFS之后,那么树的每个节点就具有了区间的性质。

     那么此时,每个节点对应了一个区间,而且可以看到,每个节点对应的区间正好“管辖”了它子树所有节点的区间,那么对点或子树的操作就转化为了对区间的操作。

Step 3:

     这个时候,每次对节点进行更新或者查询,就是线段树和树状数组最基本的实现了…

代码:

include<stdio.h>

include

include

using namespace std;

vectorv[100005];

int le[100005];

int ri[100005];

int bj[100005];

int e[100005];

int Apple[100005];

int cnt,n;



void dfs(int id)///相当于给树中的每一个节点找到合适的区间

{

int op;

le[id]=++cnt;

bj[id]=1;

for(int i=0;i<v[id].size();i++)

{

op=v[id][i];

if(bj[op]0)

{

dfs(op);

}

}

ri[id]=cnt;

}



int lowBit(int n)

{

return n&(-n);

}



int sum(int n)///求和

{

int num=0;

while(n>0)

{

num+=e[n];

n=n-lowBit(n);

}

return num;

}



void Updata(int num,int a)///更新

{

while(num<=n)

{

e[num]+=a;

num+=lowBit(num);

}

}

int main()

{

int a,b,m;

char ch;

scanf("%d",&n);

for(int i=1;i<n;i++)

{

scanf("%d%d",&a,&b);

v[a].push_back(b);

v[b].push_back(a);

}

cnt=0;

dfs(1);

for(int i=1;i<=n;i++)

{

Apple[i]=1;///刚开始的时候每个节点都有一个苹果

Updata(i,1);///初始化的时候树中的每个节点都要更新

}

scanf("%d",&m);

while(m--)

{

scanf(" %c%d",&ch,&a);

if(ch'Q')

{

printf("%d\n",sum(ri[a])-sum(le[a]-1));

}

if(ch'C')

{

if(Apple[a]1)

{

Apple[a]=0;

Updata(le[a],-1);///节点上的苹果去掉,涉及它的每个区间上都要减掉一个

}

else

{

Apple[a]=1;

Updata(le[a],1);///节点上添加一个新的苹果,涉及它的每个区间上也要加上一个苹果

}

}

}

return 0;

}

NYOJ 231 Apple Tree (树状数组)的更多相关文章

  1. POJ 3321 Apple Tree(树状数组)

                                                              Apple Tree Time Limit: 2000MS   Memory Lim ...

  2. POJ--3321 Apple Tree(树状数组+dfs(序列))

    Apple Tree Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 22613 Accepted: 6875 Descripti ...

  3. POJ 3321:Apple Tree 树状数组

    Apple Tree Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 22131   Accepted: 6715 Descr ...

  4. E - Apple Tree(树状数组+DFS序)

    There is an apple tree outside of kaka's house. Every autumn, a lot of apples will grow in the tree. ...

  5. POJ 3321 Apple Tree 树状数组+DFS

    题意:一棵苹果树有n个结点,编号从1到n,根结点永远是1.该树有n-1条树枝,每条树枝连接两个结点.已知苹果只会结在树的结点处,而且每个结点最多只能结1个苹果.初始时每个结点处都有1个苹果.树的主人接 ...

  6. POJ3321 Apple Tree(树状数组)

    先做一次dfs求得每个节点为根的子树在树状数组中编号的起始值和结束值,再树状数组做区间查询 与单点更新. #include<cstdio> #include<iostream> ...

  7. POJ 3321 Apple Tree (树状数组+dfs序)

    题目链接:http://poj.org/problem?id=3321 给你n个点,n-1条边,1为根节点.给你m条操作,C操作是将x点变反(1变0,0变1),Q操作是询问x节点以及它子树的值之和.初 ...

  8. POJ 3321 Apple Tree 树状数组 第一题

    第一次做树状数组,这个东西还是蛮神奇的,通过一个简单的C数组就可以表示出整个序列的值,并且可以用logN的复杂度进行改值与求和. 这道题目我根本不知道怎么和树状数组扯上的关系,刚开始我想直接按图来遍历 ...

  9. 3321 Apple Tree 树状数组

    LIANJIE:http://poj.org/problem?id=3321 给你一个多叉树,每个叉和叶子节点有一颗苹果.然后给你两个操作,一个是给你C清除某节点上的苹果或者添加(此节点上有苹果则清除 ...

随机推荐

  1. Halcon 笔记1

    Halcon Example位置: C:\Users\Public\Documents\MVTec\HALCON-13.0\examples 安装位置:C:\Program Files\MVTec\H ...

  2. 【EF】EF Code First Migrations数据库迁移

    1.EF Code First创建数据库 新建控制台应用程序Portal,通过程序包管理器控制台添加EntityFramework. 在程序包管理器控制台中执行以下语句,安装EntityFramewo ...

  3. HDU4045_Machine scheduling

    题意为要你从编号为1-n的所有机器中间选择出r个机器且每一个机器的编号只差不小于k-1,然后将选择的r个机器分为m组有多少种方案. 其实这题目的两个步骤是相互独立的. 总共的方案数等于选择的方案数乘以 ...

  4. poj2914-Minimum Cut

    题意 \(n\) 个点 \(m\) 条边的无向带权图求全局最小割.\(n\le 500,m\le \frac{n(n-1)}{2}\) . 分析 参考了 这篇博客,去给他点赞. 嘛,今天研究了一下全局 ...

  5. NoSQL - Redis应用场景

         问题的引入 DB(Oracle.MySQL.Postgresql等)+Memcached 这种架构模式在我们生产环境中十分常见,一般我们通过Memcached将热点数据加载到cache,应用 ...

  6. 如何在Word中排出漂亮的代码

    引言 学数学和计算机,当然还是用LaTeX排版技术文章更方便.但有时候还是迫不得已需要用Word写作,另外Word其实也有Word的好处,比如细节上的修改要比LaTeX方便. 从Matlab高亮代码复 ...

  7. 【Java】常用POI生成Excel文档设置打印样式

    package poi_test; import java.io.FileOutputStream; import java.io.IOException; import org.apache.poi ...

  8. [十二]SpringBoot 之 servlet

    Web开发使用 Controller 基本上可以完成大部分需求,但是我们还可能会用到 Servlet.Filter.Listener.Interceptor 等等. 当使用spring-Boot时,嵌 ...

  9. 不使用java内置函数,将String字符串转换为int类型

    package com.test; public class AtoiTest { public static void main(String[] args) throws Exception { ...

  10. 【NuGet】使用NuGet打包并发布至ProGet过程 (步骤详细,附python脚本)【上篇】

    一.基本知识 (1)NuGet : NuGet是一个为大家所熟知的Visual Studio扩展,通过这个扩展,开发人员可以非常方便地在Visual Studio中安装或更新项目中所需要的第三方组件, ...