Kay and Snowflake CodeForces - 686D

题意:给一棵有根树,有很多查询(100000级别的),查询是求以任意一点为根的子树的任意重心。

方法很多,但是我一个都不会

重心几个定义/性质:

1.从树中去掉某点以及和该点相连的所有边后,整棵树变为许多"块"。去掉任意一个重心(相比于去掉其他点)可以使得生成的各个"块"的节点数的最大值最少。

类似的一个:对于一棵树n个节点的无根树,找到一个点,使得把树变成以该点为根的有根树时,最大子树的结点数最小。

2.

(1) 树中所有点到某个点的距离和中,到重心的距离和是最小的;如果有两个重心,那么他们的距离和一样。

(2) 把两个树通过一条边相连得到一个新的树,那么新的树的重心在连接原来两个树的重心的路径上。

(3) 把一个树添加或删除一个叶子,那么它的重心最多只移动一条边的距离。

3.(就是题面所讲的)从树中去掉某点以及和该点相连的所有边后,整棵树变为许多"块"。

去掉任意一个重心后,生成的各个"块"的节点数的最大值一定小于等于原树的节点数除以2。

4.以一棵树的重心为根的子树的节点个数,一定大于等于该树节点总数的一半。(因为不然的话把它去掉那么它的上面产生的那个连通块内部节点个数一定大于总节点数的一半,不符合性质3)

5.一棵树一定存在重心。

6.在一棵树的所有子树中,找到某一子树,使得其节点数恰好大于等于原树节点总数一半(也就是满足"节点数大于等于原树节点总数一半"这个条件的子树中节点数最小的子树),那么该子树的根一定是一个重心。

(如果该节点不是重心,也就是把它去掉后产生的连通块中至少有一个节点个数大于原树节点个数的一半。

显然这个连通块不是它上面产生的连通块,因为"(该子树)节点数大于等于原树节点总数一半",也就是它上面产生的连通块节点数一定小于原树节点总数一半。

也不可能是下面的连通块,因为它是满足"节点数大于等于原树节点总数一半"这个条件的子树中节点数最小的子树,任何节点数小于它的子树的节点数都小于原树节点总数一半,而以其子节点为根的子树的节点数一定小于以其为根的子树。

这样就产生了矛盾,因此得证

http://codeforces.com/blog/entry/45558

关于启发式合并:

http://codeforces.com/blog/entry/21827中E的题解。

合并两个答案的集合(可以为set,手写平衡树等)时,始终把size小的合并到size大的上面(相等size则任意)。

那么对于某一类问题,考虑某一个元素,会发现每次它被从一个集合合并到另一个集合时,它所在集合size至少扩大到了原来的二倍。如果最终答案集合的size是n,有n个元素,那么合并总次数就不会超过n log n级别。

方法:

1.

在某有根树中,如果(以根节点的某一子节点v为根的子树)是(以根节点的任一子节点为根的所有子树)中节点数量最多的,那么称v为该有根树根节点的最重子节点。

可以发现,一棵有根树的重心,一定在根节点到(以其最重子节点为根的子树)的重心的路径上。(根据性质1感性理解吧...)

问题:怎么证?

因此,先预处理出以任意节点为根的子树的节点数。对于某一节点,可以找到最重子节点,然后暴力从对应子树的重心往上跳枚举新重心。根据性质3,可以直接判断一个节点是不是某子树的重心,不需要通过比较。每条边最多被跳一次,边数=点数-1,因此复杂度O(n)。

2.

利用性质4、5、6以及启发式合并思想,dfs每个点,每个点返回一个包含其中所有子树size和编号的set。这样元素被加入set次数不超过n log n级别,每次加入复杂度为log n,总复杂度O(n log^2 n)。

http://codeforces.com/blog/entry/21827==>600E - Lomsat gelral  "merge sets" idea

 #include<cstdio>
#include<algorithm>
#include<set>
using namespace std;
typedef pair<int,int> P;
struct E
{
int to,nxt;
}e[];
int f1[],ne,ans[],n,q;
set<P> dfs(int u,int fa)
//返回该子树中各个子树的size组成的集合
//P(size,编号)
{
set<P> tmp,t;
for(int k=f1[u];k;k=e[k].nxt)
if(e[k].to!=fa)
{
t=dfs(e[k].to,u);
if(t.size()>tmp.size()) swap(t,tmp);
tmp.merge(t);
}
int sz=tmp.size()+;
tmp.emplace(sz,u);
ans[u]=tmp.lower_bound(P((sz+)/,))->second;
return tmp;
}
int main()
{
int i,t;
scanf("%d%d",&n,&q);
for(i=;i<=n;i++)
{
scanf("%d",&t);
e[++ne].to=t;e[ne].nxt=f1[i];f1[i]=ne;
e[++ne].to=i;e[ne].nxt=f1[t];f1[t]=ne;
}
dfs(,);
while(q--)
{
scanf("%d",&t);
printf("%d\n",ans[t]);
}
return ;
}

Kay and Snowflake CodeForces - 686D的更多相关文章

  1. Kay and Snowflake CodeForces - 686D (树的重心性质)

    After the piece of a devilish mirror hit the Kay's eye, he is no longer interested in the beauty of ...

  2. Kay and Snowflake CodeForces - 685B (重心, 好题)

    大意:给定有根树, 求每个子树的重心 我太菜了啊, 只能想到暴力树剖, 然而这就是个B题, 感觉树剖+线段树二分还是挺难写的..... 看了题解发现重心一定在重儿子与根的树链上, 重心最多上跳n-1次 ...

  3. codeforces 686D D. Kay and Snowflake(dfs)

    题目链接: D. Kay and Snowflake time limit per test 3 seconds memory limit per test 256 megabytes input s ...

  4. Codeforces Round #359 (Div. 1) B. Kay and Snowflake dfs

    B. Kay and Snowflake 题目连接: http://www.codeforces.com/contest/685/problem/B Description After the pie ...

  5. Codeforces Round #359 (Div. 2) D. Kay and Snowflake 树DP

    D. Kay and Snowflake     After the piece of a devilish mirror hit the Kay's eye, he is no longer int ...

  6. Codeforces Round #359 (Div. 2) D. Kay and Snowflake 树的重心

    题目链接: 题目 D. Kay and Snowflake time limit per test 3 seconds memory limit per test 256 megabytes inpu ...

  7. Codeforces Round #359 (Div. 2) D - Kay and Snowflake

    D - Kay and Snowflake 题目大意:给你一棵数q个询问,每个询问给你一个顶点编号,要你求以这个点为根的子树的重心是哪个节点. 定义:一棵树的顶点数为n,将重心去掉了以后所有子树的顶点 ...

  8. Codeforces 686 D - Kay and Snowflake

    D - Kay and Snowflake 思路: 树的重心 利用重心的一个推论,树的重心必定在子树重心的连线上. 然后利用重心的性质,可知,如果有一颗子树的大小超过整棵树的大小的1/2,那么树的重心 ...

  9. CF685B Kay and Snowflake 贪心

    CF685B Kay and Snowflake 链接 CF 题目大意 给你一颗树,询问子树的重心 思路 贪心? 重心肯定是向上走的,所以直接向上跳就好了. 不优秀的时候就不要跳了 ,因为以后也不能更 ...

随机推荐

  1. 利用百度地图Android sdk高仿微信发送位置功能

    接触了百度地图开发平台半个月了,这2天试着模仿了微信给好友发送位置功能,对百度地图的操作能力又上了一个台阶 (假设须要完整demo.请评论留下邮箱) (眼下源代码已经不发送,假设须要源代码.加qq31 ...

  2. 【stl学习笔记】红黑树

    转自维基百科 红黑树是一种平衡二叉搜索树,它可以在O(log n)时间内做查找,插入和删除,这里的n是树中元素的数目. 性质: 红黑树是每个节点都带有颜色属性的二叉查找树,颜色为红色或黑色.在二叉查找 ...

  3. [原创+分享]Mandelbrot Explorer

    Mandelbrot Explorer 是一款用于在MandelBort集/Julia集上进行无限漫游的软件,使用VS2013+CUDA6.5开发而成.它也是我学习CUDA开发的一个小小的成果,欢迎大 ...

  4. 嵌入式开发之davinci--- 8148/8168/8127 中的图像缩放sclr、swms之后出现图像视频卡顿、屏幕跳跃的问题

    ()问题原因 这边的case链路是这样的camera->sclr(yuv420sp cif)->dup->ipcframeoutm3<->ipcframerocess&l ...

  5. 多媒体开发之wis-stream

    在live555的mediaServer中,已经实现RTSP-over-HTTP,但默认没有开启.如果要实现这个功能,需要调用RTSPServer::setUpTunnelingOverHTTP(), ...

  6. 堆排序C++实现

    //heap sort //堆排序能够分为两个过程.其一是建堆.其二是出堆 //堆是一种全然二叉树,所以它能够用数组进行存储. //堆可分为最大堆和最小堆.最大堆指任一节点的值都大于其左右孩子节点的值 ...

  7. Blocks实现代理传值

    一.RootViewController: #import "RootViewController.h" #import "SecondViewController.h& ...

  8. 在Eclipse中tomcat 内存配置

    修改1: 在Eclipse中下面Servers双击Tomcat Server... 然后点击General InformAtion 下的Open launch configuration: 会弹出Ed ...

  9. HDU3038 How Many Answers Are Wrong —— 带权并查集

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=3038 How Many Answers Are Wrong Time Limit: 200 ...

  10. YTU 2801: 用数字造数字(II)

    2801: 用数字造数字(II) 时间限制: 1 Sec  内存限制: 128 MB 提交: 244  解决: 168 题目描述 输入一个3位以上的整数,求其中最大的两个数字之和与最小的数字之和之间的 ...