题目描述

武当派一共有 n 人,门派内 n 人按照武功高低进行排名,武功最高的人排名第 1,次高的人排名第 2,... 武功最低的人排名
第 n。现在我们用武功的排名来给每个人标号,除了祖师爷,每个人都有一个师父,每个人可能有多个徒弟。
我们知道,武当派人才辈出,连祖师爷的武功都只能排行到 pp。也就是说徒弟的武功是可能超过师父的,所谓的青出于蓝胜于蓝。
请你帮忙计算每个人的所有子弟(包括徒弟的徒弟,徒弟的徒弟的徒弟....)中,有多少人的武功超过了他自己。

输入格式

输入第一行两个整数 n, p(1 ≤ n ≤ 100000, 1 ≤  p ≤ n) n , p (1≤n≤100000,1≤p≤n)。
接下来 n-1 行,每行输入两个整数 u, v(1 ≤ u, v ≤ n)u , v  (1≤u,v≤n),表示 u 和 v 之间存在师徒关系。

输出格式

输出一行 n 个整数,第 i 个整数表示武功排行为 i 的人的子弟有多少人超过了他。
行末不要输出多余的空格。

样例输入

10 5

5 3

5 8

3 4

3 1

2 1

6 7

8 7

9 8

8 10

样例输出

0 0 2 0 4 0 1 2 0 0

分析

①将每条边输入进去,然后dfs求出这棵树的dfs序,在求dfs序的过程中,数组 num2 记录每个节点的孩子节点数量。

正如样例输入,dfs序为 5,3,4,1,2,8,7,6,9,10

②遍历我们找到的dfs序,数组 d 记录每个编号在dfs序中的位置,为了统一处理,我的dfs序列是从1开始计数的,而不是0。

这一步是为了将这个树形结构序列化,方便我们进行求和操作

③遍历数组d,num数组记录武功排行为 i 的人的子弟有多少人超过了他。

           a.   num [ i ] = getsum( d [ i ] + num2[ i ] ) – getsum ( d [ i ] )

由于我们我们是从小到大枚举 i ,因此 getsum( d [ i ] + num[ i ] ) – getsum ( d [ i ] ) 就是满足条件的子弟数量。

           b.   change( d[ i ], 1)

观察第三步中的a步骤,我们发现,num2[i]在求出num[i]后再也没用过,因此num2 与 num 可以共用一个储存空间。

代码:

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <vector>
#include <stdlib.h>
#include <string.h>
#define MAX 100000
using namespace std;
vector<int> edge[MAX];
int n, root;
int cnt = 1;//记录个数,用于构建dfs序
bool vis[MAX];//记录每条边是否访问过,用于构建dfs序
int dfs_seq[MAX];//记录dfs序
int c[MAX];//树状数组
int num[MAX];//记录每个编号有多少个满足要求的徒弟
int d[MAX];//记录每个节点在dfs序中的位置
void add(int a, int b){
edge[b].push_back(a);
edge[a].push_back(b);
} //树状数组操作
int lowbit(int x){
return x & (-x);
} void change(int x, int v){
while(x <= n){
c[x] += v;
x += lowbit(x);
}
} int getsum(int x){
int sum = 0;
while(x > 0){
sum += c[x];
x -= lowbit(x);
}
return sum;
}
//构建dfs序
int dfs(int root){
int sum = 0;
vis[root] = 1;
dfs_seq[cnt++] = root;
//cout << dfs_seq[cnt - 1] << " ";
for(int i = 0; i < edge[root].size(); i++){
if(!vis[edge[root][i]])
sum += dfs(edge[root][i]) + 1;
}
num[root] = sum;
return sum;
} int main() {
scanf("%d %d", &n, &root);
for(int i = 1; i < n; i++){
int a, b;
scanf("%d %d", &a, &b);
//将此边加入
add(a, b);
} dfs(root); //记录每个节点在dfs序中的位置
for(int i = 1; i <= n; i++){
d[dfs_seq[i]] = i;
} for(int i = 1; i <= n; i++){
num[i] = getsum(d[i] + num[i]) - getsum(d[i]);
change(d[i], 1);
} //依次输出
for(int i = 1; i <= n; i++){
cout << num[i] << "";
}
return 0;
}

计蒜客 青出于蓝胜于蓝(dfs序+树状数组)的更多相关文章

  1. 计蒜客 31451 - Ka Chang - [DFS序+树状数组][2018ICPC沈阳网络预赛J题]

    题目链接:https://nanti.jisuanke.com/t/31451 Given a rooted tree ( the root is node $1$ ) of $N$ nodes. I ...

  2. 计蒜客A1998 Ka Chang (分块+dfs序+树状数组)

    题意 给你一个\(1e5\)的有点权的树,有\(1e5\)个操作: 1.给第\(x\)层的点加上\(y\) 2.求以\(x\)为根的子树的点权和 思路 首先处理出层数为x的所有点 操作2一般都是用df ...

  3. HDU 3887:Counting Offspring(DFS序+树状数组)

    http://acm.hdu.edu.cn/showproblem.php?pid=3887 题意:给出一个有根树,问对于每一个节点它的子树中有多少个节点的值是小于它的. 思路:这题和那道苹果树是一样 ...

  4. HDU 5293 Tree chain problem 树形dp+dfs序+树状数组+LCA

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5293 题意: 给你一些链,每条链都有自己的价值,求不相交不重合的链能够组成的最大价值. 题解: 树形 ...

  5. Codeforces Round #225 (Div. 1) C. Propagating tree dfs序+树状数组

    C. Propagating tree Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/383/p ...

  6. BZOJ 2434: [Noi2011]阿狸的打字机( AC自动机 + DFS序 + 树状数组 )

    一个串a在b中出现, 那么a是b的某些前缀的后缀, 所以搞出AC自动机, 按fail反向建树, 然后查询(x, y)就是y的子树中有多少是x的前缀. 离线, 对AC自动机DFS一遍, 用dfs序+树状 ...

  7. 【bzoj3881】[Coci2015]Divljak AC自动机+树链的并+DFS序+树状数组

    题目描述 Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: “1 P”,Bob往自己的集合里添加了一个字符串P. ...

  8. [BZOJ1103][POI2007]大都市meg dfs序+树状数组

    Description 在经济全球化浪潮的影响下,习惯于漫步在清晨的乡间小路的邮递员Blue Mary也开始骑着摩托车传递邮件了.不过,她经常回忆起以前在乡间漫步的情景.昔日,乡下有依次编号为1..n ...

  9. 2018.10.20 NOIP模拟 巧克力(trie树+dfs序+树状数组)

    传送门 好题啊. 考虑前面的32分,直接维护后缀trietrietrie树就行了. 如果#号不在字符串首? 只需要维护第一个#前面的字符串和最后一个#后面的字符串. 分开用两棵trie树并且维护第一棵 ...

随机推荐

  1. 20200227英语上课笔记 about advantage and disadvantage

    Hello and welcome to class! Remember to keep your microphone off when you are not speaking Pronuncia ...

  2. HttpServer带阻塞性能比较

    服务端在返回hello,world! 之前先阻塞一秒钟,模拟访问DB等耗时操作. Netty 我直接在 WorkerGroup 里头sleep,用同步阻塞线程模型的方式来编程,所以性能暴降. Joob ...

  3. rabbitmq - 简单认识

    1. 概述 与 rabbitmq 做交互 amqp 最著名的实现 与 jms 最明显的区别 消息 不是去找 queue 而是去找 exchange 2. rabbitmq 基本组件 sender 发送 ...

  4. TCL Strings

    append    Append values to variable binary      Insert and extract fields from binary strings regexp ...

  5. Bugku-CTF加密篇之奇怪的密码(突然天上一道雷电 gndk€rlqhmtkwwp}z )

    奇怪的密码 突然天上一道雷电 gndk€rlqhmtkwwp}z  

  6. 试题编号: 201903-3 试题名称: 损坏的RAID5

    这题的数据未免也太水了,题目的意思好像默认是每块磁盘装载数据的长度是相等的.我写了判断每次取数据是否会超过每块磁盘存的数据的长度,然而并没有什么卵用.交上去20分,写了个数据测了下,如果要求的块太大的 ...

  7. NLP开源工具

    最近有人问我几次NLP有哪些开源工具,这里做个笔记.

  8. pycharm新建Django时,遇到的坑,安装index包失败

    https://blog.csdn.net/li93675/article/details/89418097 如果在pycharm中导入django包 ,只对当前项目有效,建议使用命令pip inst ...

  9. C语言:使用递归解决汉诺塔问题。

    //汉诺塔:汉诺塔(又称河内塔)问题是源于印度一个古老传说的益智玩具.大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘.大梵天命令婆罗门把圆盘从下面开始按大小 ...

  10. 《MFC dialog中加入OpenGL窗体》

    <MFC dialog中加入OpenGL窗体> 最近学习了如何在MFC对话框程序中加入OpenGL窗体的方法,在这里将自己的实现过程归纳一下. 步骤零: 加入PictureControl控 ...