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. HDU1087 Super Jumping! Jumping! Jumping!(LIS)

    题目意思: http://acm.hdu.edu.cn/showproblem.php? pid=1087 此题的意思求最长上升子序列的和. 题目分析: 在求最长上升子序列的时候,不在保存最长的个数, ...

  2. 【转】c++内存管理学习纲要

    http://blog.csdn.net/zhanghefu/article/details/5003407 转自:http://blog.csdn.net/wdzxl198/article/deta ...

  3. POJ2752 Seek the Name, Seek the Fame 【KMP】

    Seek the Name, Seek the Fame Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 11602   Ac ...

  4. [TypeScript] Query Properties with keyof and Lookup Types in TypeScript

    The keyof operator produces a union type of all known, public property names of a given type. You ca ...

  5. MYSQL将时间格式化

    SELECT *,DATE_FORMAT(FROM_UNIXTIME(createtime), "%Y/%m/%d %H:%i:%s") AS dateFormat FROM `I ...

  6. 【转】TestNG常用注解

    http://blog.csdn.net/d6619309/article/details/52435084 TestNG的注解大部分用在方法级别上.常用的注解列举如下: 1. Before类别和Af ...

  7. icvSetWeightsAndClasses

    /* *icvSetWeightsAndClasses *作用:给训练样本的权重和类别赋值 */ static void icvSetWeightsAndClasses( CvHaarTraining ...

  8. hdu1133 Buy the Ticket (卡兰特数应用+java大数)

    题目链接:http://acm.hdu.edu.cn/showproblem.php? pid=1133 [题意] 电影票50块一张 有m个人手里正好有50块,n个人手里正好有100块,售票厅開始没有 ...

  9. 关于 iOS 的 StoryBoard,接受的那一刻才发现她的美 - 当然美的事物都须要业心照料

    关于 iOS 的 StoryBoard,接受的那一刻才发现她的美 - 当然美的事物都须要业心照料 太阳火神的漂亮人生 (http://blog.csdn.net/opengl_es) 本文遵循&quo ...

  10. c语言学习-指针探究

    1:指针定义格式:格式:变量类型 *变量名用途:指针变量用于储存地址(only),也就是根据地址值,访问对应的存储空间. 注意.int *p 只能指向int类型的数据: 例: int a = 20; ...