[luogu5666]树的重心
考虑枚举一个点k,求其为重心的方案数
暴力的做法是,将其作为根搜索,设最大子树大小为s1,次大为s2,对割掉的子树分类讨论:
1.在子树中,分两种情况(都可以用线段树合并来做)
(1)从s1中切掉一棵大小为s3的子树,应该满足$2max(s2,s1-s3)\le n-s3$,即$2s1-n\le s3\le n-2s2$
(2)从其他子树中切掉一棵大小为s3的子树,应该满足$2s1\le n-s3$,即$s3\le n-2s1$
2.是父亲,那么割掉的边再分为两类(这些东西也需要再根据父亲是不是最大子树来讨论)
(1)割掉的边是直接到根的路径,那么割掉的子树大小s3就是n-割出来的子树,可以再搜一遍不断的维护当前节点到根的路径上所有子树大小,用权值线段树来维护区间和
(2)割掉的边是其他边,直接对最终线段树合并到根的线段树上查询即可(注意这样会错误计算第(1)种情况,要注意在第一个中抵消掉,即再对子树大小打上-1标记)

1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 300005
4 #define mid (l+r>>1)
5 struct ji{
6 int nex,to;
7 }edge[N<<1];
8 struct node{
9 int sum,ls,rs;
10 }f[N*50];
11 int E,V,t,n,x,y,head[N],r[N],fi[N],se[N],sz[N];
12 long long ans;
13 void up(int k){
14 f[k].sum=f[f[k].ls].sum+f[f[k].rs].sum;
15 }
16 void update(int &k,int l,int r,int x,int y){
17 if (!k)k=++V;
18 if (l==r){
19 f[k].sum+=y;
20 return;
21 }
22 if (x<=mid)update(f[k].ls,l,mid,x,y);
23 else update(f[k].rs,mid+1,r,x,y);
24 up(k);
25 }
26 int query(int k,int l,int r,int x,int y){
27 if ((!k)||(l>y)||(x>r))return 0;
28 if ((x<=l)&&(r<=y))return f[k].sum;
29 return query(f[k].ls,l,mid,x,y)+query(f[k].rs,mid+1,r,x,y);
30 }
31 int merge(int k1,int k2){
32 if ((!k1)||(!k2))return k1+k2;
33 if ((!f[k1].ls)&&(!f[k1].rs)){
34 f[k1].sum+=f[k2].sum;
35 return k1;
36 }
37 f[k1].ls=merge(f[k1].ls,f[k2].ls);
38 f[k1].rs=merge(f[k1].rs,f[k2].rs);
39 up(k1);
40 return k1;
41 }
42 void add(int x,int y){
43 edge[E].nex=head[x];
44 edge[E].to=y;
45 head[x]=E++;
46 }
47 void dfs(int k,int fa){
48 fi[k]=sz[k]=0;
49 sz[k]=1;
50 for(int i=head[k];i!=-1;i=edge[i].nex)
51 if (edge[i].to!=fa){
52 dfs(edge[i].to,k);
53 sz[k]+=sz[edge[i].to];
54 if (sz[edge[i].to]<fi[k])se[k]=max(se[k],sz[edge[i].to]);
55 else{
56 se[k]=fi[k];
57 fi[k]=sz[edge[i].to];
58 }
59 }
60 if (n-sz[k]<fi[k])se[k]=max(se[k],n-sz[k]);
61 else{
62 se[k]=fi[k];
63 fi[k]=n-sz[k];
64 }
65 for(int i=head[k];i!=-1;i=edge[i].nex)
66 if (edge[i].to!=fa){
67 if (sz[edge[i].to]!=fi[k])ans+=1LL*k*query(r[edge[i].to],1,n,1,n-2*fi[k]);
68 else ans+=1LL*k*query(r[edge[i].to],1,n,max(2*fi[k]-n,1),n-2*se[k]);
69 r[k]=merge(r[k],r[edge[i].to]);
70 }
71 if (fi[k]!=n-sz[k])ans-=1LL*k*query(r[k],1,n,1,n-2*fi[k]);
72 else ans-=1LL*k*query(r[k],1,n,max(2*fi[k]-n,1),n-2*se[k]);
73 update(r[k],1,n,sz[k],1);
74 }
75 void dfs2(int k,int fa){
76 update(r[1],1,n,sz[k],-1);
77 if (k>1)update(r[1],1,n,n-sz[k],1);
78 if (fi[k]!=n-sz[k])ans+=1LL*k*query(r[1],1,n,1,n-2*fi[k]);
79 else ans+=1LL*k*query(r[1],1,n,max(2*fi[k]-n,1),n-2*se[k]);
80 for(int i=head[k];i!=-1;i=edge[i].nex)
81 if (edge[i].to!=fa)dfs2(edge[i].to,k);
82 update(r[1],1,n,sz[k],1);
83 update(r[1],1,n,n-sz[k],-1);
84 }
85 int main(){
86 scanf("%d",&t);
87 while (t--){
88 E=V=ans=0;
89 memset(r,0,sizeof(r));
90 memset(f,0,sizeof(f));
91 memset(head,-1,sizeof(head));
92 scanf("%d",&n);
93 for(int i=1;i<n;i++){
94 scanf("%d%d",&x,&y);
95 add(x,y);
96 add(y,x);
97 }
98 dfs(1,0);
99 dfs2(1,0);
100 printf("%lld\n",ans);
101 }
102 }
[luogu5666]树的重心的更多相关文章
- POJ3107Godfather[树形DP 树的重心]
Godfather Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 6121 Accepted: 2164 Descrip ...
- poj1655 树的重心 树形dp
树的重心定义为:找到一个点,其所有的子树中最大的子树节点数最少,那么这个点就是这棵树的重心,删去重心后,生成的多棵树尽可能平衡. 处理处每个节点的孩子有几个,和树的大小就好了. #include< ...
- poj3107 求树的重心(&& poj1655 同样求树的重心)
题目链接:http://poj.org/problem?id=3107 求树的重心,所谓树的重心就是:在无根树转换为有根树的过程中,去掉根节点之后,剩下的树的最大结点最小,该点即为重心. 剩下的数的 ...
- 树形DP求树的重心 --SGU 134
令一个点的属性值为:去除这个点以及与这个点相连的所有边后得到的连通分量的节点数的最大值. 则树的重心定义为:一个点,这个点的属性值在所有点中是最小的. SGU 134 即要找出所有的重心,并且找出重心 ...
- 求树的重心(POJ1655)
题意:给出一颗n(n<=2000)个结点的树,删除其中的一个结点,会形成一棵树,或者多棵树,定义删除任意一个结点的平衡度为最大的那棵树的结点个数,问删除哪个结点后,可以让平衡度最小,即求树的重心 ...
- codeforces 685B Kay and Snowflake 树的重心
分析:就是找到以每个节点为根节点的树的重心 树的重心可以看这三篇文章: 1:http://wenku.baidu.com/link?url=yc-3QD55hbCaRYEGsF2fPpXYg-iO63 ...
- POJ 1655 Balancing Act (求树的重心)
求树的重心,直接当模板吧.先看POJ题目就知道重心什么意思了... 重心:删除该节点后最大连通块的节点数目最小 #include<cstdio> #include<cstring&g ...
- POJ3107--Godfather(树的重心)
vector建图被卡了..改为链式前向星500ms过的..差了四倍多?... 表示不太会用链表建图啊..自己试着写的,没看模板..嗯..果然错了..落了一句话orz 树的重心就是找到一个树中一个点,其 ...
- POJ 1655 Balancing Act&&POJ 3107 Godfather(树的重心)
树的重心的定义是: 一个点的所有子树中节点数最大的子树节点数最小. 这句话可能说起来比较绕,但是其实想想他的字面意思也就是找到最平衡的那个点. POJ 1655 题目大意: 直接给你一棵树,让你求树的 ...
随机推荐
- appium启动ios系统上面的app需求的参数
Appium启动APP至少需要7个参数 'platformVersion','deviceName'.'udid'.'bundleId'.'platformName'.'automationName ...
- [JUC-5]ConcurrentHashMap源码分析JDK8
在学习之前,最好先了解下如下知识: 1.ReentrantLock的实现和原理. 2.Synchronized的实现和原理. 3.硬件对并发支持的CAS操作及JVM中Unsafe对CAS的实现. 4. ...
- 【UE4 C++】 UnrealPak 与 Pak 的制作、挂载、加载
简介 通过 UnrealPak,可以将资源打包成 Pak 文件 Pak文件是UE4游戏生成的数据包文件. Pak 之前一般先有 Cooked 步骤,将资源烘焙为对应平台支持的资源 一般打包后的项目使用 ...
- sql递归查询部门数据
1 with cte as 2 ( 3 select a.DepartCode,a.DepartName,a.ParentDepartCode from tbDeparts a where Paren ...
- 第5次 Beta Scrum Meeting
本次会议为Beta阶段第6次Scrum Meeting会议 会议概要 会议时间:2021年6月6日 会议地点:「腾讯会议」线上进行 会议时长:10min 会议内容简介:对完成工作进行阶段性汇报:对下一 ...
- linux与windows下文件编码问题
注:转换操作均在Linux终端进行操作 DOS与Unix格式转换 安装工具:dos2unix.unix2dos # ubuntu apt-get install dos2unix apt-get in ...
- Python中的括号()、[]、{}
长时间不用容易混淆,仅记! 在Python语言中最常见的括号有三种,分别是:小括号().中括号[].花括号{} . Python中的小括号(): 代表tuple元祖数据类型,元祖是一种不可变序列.大多 ...
- Vulnhub实战-dr4g0n b4ll靶机👻
Vulnhub实战-dr4g0n b4ll靶机 地址:http://www.vulnhub.com/entry/dr4g0n-b4ll-1,646/ 描述:这篇其实没有什么新奇的技巧,用到的提权方式就 ...
- IDA*、操作打表、并行处理-The Rotation Game HDU - 1667
万恶之源 优秀题解 用文字终究难以穷尽代码的思想 思路 每次操作都有八种选择,相当于一棵每次延申八个子节点的搜索树,故搜索应该是一种方法.而这题要求求最少步数,我们就可以想到可以试试迭代加深搜索(但其 ...
- 利用Ambari平台安装与部署Hadoop
* 本篇是利用Ambari平台安装与部署Hadoop,如果需要原生部署Hadoop,请点击以下地址: https://www.cnblogs.com/live41/p/15467263.html 一. ...