[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 题目大意: 直接给你一棵树,让你求树的 ...
随机推荐
- 搭建Mac+Java+appium+IOS真机自动化环境
一.安装前环境准备 1.确保电脑已经有homebrew(包管理器) 下载链接[https://brew.sh/] 2.通过 brew 安装node.js brew install node 安装 ...
- MyBatis 中实现SQL语句中in的操作 (11)
MyBatis 中实现SQL语句中in的操作 概括:应用myBatis实现SQL查询中IN的操作 1.数据库结构及其数据 2.mapper.xml文件 <?xml version="1 ...
- Mybatis 一对多延迟加载,并且子查询中与主表字段不对应 (19)
Mybatis 一对多延迟加载,并且子查询中与主表字段不对应应用说明. 实现一对多关联(懒加载),一个教研组对应多个教师,既:教师的教研编号与教研组的教研编号关联,并且教师关联教研组外键与教研组编号 ...
- gitk
gitk gitk [<options>] [<revision range>] [--] [<path>-] 查看单个文件的变更历史 gitk -- CppPri ...
- spring源码分析(二)- 容器基础
1.基本用法 用过Spring的都知道,bean是Spring中最基础也是最核心的.首先看一个简单的例子. 一个类和一个配置文件 package bean; public class MyBean { ...
- 2020年OO助教工作总结
随着这学期课程的落幕,我一学期的OO助教工作也宣告结束.这学期我的工作主要在系统组,和OO后台的数据库打交道. 作业查重 我几乎每周都会做的例行工作,是对每周的homework进行查重管理.由于使用了 ...
- SpringBoot整合多个RabbitMQ
一.背景 最近项目中需要用到了RabbitMQ来监听消息队列,监听的消息队列的 虚拟主机(virtualHost)和队列名(queueName)是不一致的,但是接收到的消息格式相同的.而且可能还存 ...
- Pogo-Cow S
这题出在单调队列优化dp里,就离谱好吧...... 对不住了上来先喷一波,不过离谱是确实的 dp的含义也很简单,就是说从j到i的分数最大值 直接上代马,里面说的很详细了 1 #include<b ...
- HCNP Routing&Switching之BGP路由属性和优选规则
前文我们了解了BGP防环机制和路由聚合相关话题,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/15458110.html:今天我们来聊一聊BGP路由属性和选路规 ...
- Envoy实现.NET架构的网关(四)集成IdentityServer4实现OAuth2认证
什么是OAuth2认证 简单说,OAuth 就是一种授权机制.数据的所有者告诉系统,同意授权第三方应用进入系统,获取这些数据.系统从而产生一个短期的进入令牌(token),用来代替密码,供第三方应用使 ...