这是找树的重心的经典题目。

树的重心有下面几条常见性质:

定义1:找到一个点,其所有的子树中最大的子树节点数最少,那么这个点就是这棵树的重心
定义2:以这个点为根,那么所有的子树(不算整个树自身)的大小都不超过整个树大小的一半。
性质1:树中所有点到某个点的距离和中,到重心的距离和是最小的;如果有两个重心,那么他们的距离和一样。
性质2:把两个树通过一条边相连得到一个新的树,那么新的树的重心在连接原来两个树的重心的路径上。
性质3:把一个树添加或删除一个叶子,那么它的重心最多只移动一条边的距离。

方法:就记节点1为树的根,两次dfs,第一次求出每个节点的所有子孙再加上它自己的节点总数num[i]。第二次就算出每个节点的balance值bal[i],算的时候就比较节点i它所有子节点的num值(删掉它之后以每个它的子节点为根形成一棵新树)还有n-num[i]的值(删掉i之后它的父节点及其相关节点也形成一棵新树),最大的就是bal[i]。

注意:WA了几次是因为没有考虑边界情况(n==2),dfs写的太不熟练了,代码能力有待提高啊!

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<map>
#include<set>
#include<vector>
#include<algorithm>
#include<stack>
#include<queue>
using namespace std;
#define INF 1000000000
#define eps 1e-8
#define pii pair<int,int>
#define LL long long int
int T,n,a,b,ans_i,ans;
vector<int>v[];
int num[],bal[];
int dfs1(int i,int fa);
void dfs2(int i,int fa);
int main()
{
//freopen("in2.txt","r",stdin);
//freopen("out.txt","w",stdout);
scanf("%d",&T);
while(T--)
{
ans_i=;
ans=INF;
scanf("%d",&n);
if(n==)
{
printf("1 0\n");
}
else if(n==)
{
scanf("%d%d",&a,&b);
printf("1 1\n");
}
else
{
for(int i=; i<=n; i++)
{
v[i].clear();
num[i]=;
bal[i]=-;
}
for(int i=; i<n; i++)
{
scanf("%d%d",&a,&b);
v[a].push_back(b);
v[b].push_back(a);
}
dfs1(,-);
dfs2(,-);
for(int i=; i<=n; i++)
{
if(bal[i]<ans)
{
ans_i=i;
ans=bal[i];
}
}
printf("%d %d\n",ans_i,ans);
}
}
//fclose(stdin);
//fclose(stdout);
return ;
}
int dfs1(int i,int fa)
{
/*if(v[i].size()==1)
return num[i];*/
/*这句判断是不能加的,加了就是WA,个中原因非常微妙:
因为我加这句的本意是当遇到了叶子节点时,它的num就直接返回1就行了,
而叶节点的size就是1(只有父节点)。
但是我这里就忽略了一种特殊情况,那就是这棵数可能根节点的size也为1!!!
所以就在这种情况下WA了。
实际上没有必要加这一句,下面的循环语句已经可以很好的处理各个节点了。*/
//这里可以总结一条树的性质:叶节点一定含一条边,根节点可能含一条边,其它节点至少含两条边。
for(unsigned int j=; j<v[i].size(); j++)
{
int &t=v[i][j];
if(t==fa)
{
continue;
}
else
{
num[i]+=dfs1(t,i);
}
}
//cout<<i<<"_-_"<<num[i]<<endl;
return num[i];
}
void dfs2(int i,int fa)
{
for(unsigned int j=; j<v[i].size(); j++)
{
int &t=v[i][j];
if(t==fa)
{
bal[i]=max(bal[i],num[]-num[i]);
}
else
{
bal[i]=max(bal[i],num[t]);
dfs2(t,i);
}
}
//cout<<i<<"___"<<bal[i]<<endl;
}

后来看了其它人的题解发现其实这两次的dfs结构差不多,写一次dfs就足够了,减少代码量。而且只要求最小bal,不用每个点的bal都保存,这样可以节省空间。还有一个小技巧,就是我这是从根节点遍历树,我没有必要用一个标记数组uesd[maxn]来记录每个点是不是遍历过了,只要判断每次是不是父节点就行了,父节点肯定遍历过,子节点肯定没遍历过。这样就用节省空间了。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<map>
#include<set>
#include<vector>
#include<algorithm>
#include<stack>
#include<queue>
using namespace std;
#define INF 1000000000
#define eps 1e-8
#define pii pair<int,int>
#define LL long long int
const int maxn=;
int T,n,a,b,head[maxn],cnt,ansi,ansb;
struct node
{
int v,next;
}e[maxn<<];/*前向星存图时这个结构体数组存的是边的信息,
如果是无向图,千万注意要开二倍!这里容易错。*/
void ini()
{
memset(head,-,sizeof(int)*(n+));
cnt=;
ansi=ansb=INF;
}
void add(int aa,int bb)
{
e[cnt].v=bb;
e[cnt].next=head[aa];
head[aa]=cnt++;
}
int dfs(int x,int fa)
{
int sum=,bx=,t;
//sum是以当前的x节点为根的子树所包含的节点数
//bx是x节点的平衡值
for(int i=head[x];i!=-;i=e[i].next)
{
if(e[i].v==fa) continue;/*使用前向星存图在遍历每个点的边时要注意i是边的标号而
不是点的标号,这里不能误写成i==fa*/
else
{
t=dfs(e[i].v,x);
bx=max(bx,t);//用子树的节点数更新bx
sum+=t;
}
}
bx=max(n-sum,bx);//这一步不能少
if((bx<ansb)||(bx==ansb&&ansi>x))
{
ansi=x;
ansb=bx;
}
return sum;
}
int main()
{
//freopen("in6.txt","r",stdin);
//freopen("out.txt","w",stdout);
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
ini();
for(int i=;i<=n-;i++)
{
scanf("%d%d",&a,&b);
add(a,b);
add(b,a);
}
dfs(,-);
printf("%d %d\n",ansi,ansb);
}
return ;
}

poj1655(dfs,树形dp,树的重心)的更多相关文章

  1. poj1655(dfs,树形dp,树的重心)(点分治基础)

    题意:就是裸的求树的重心. #include<cstring> #include<algorithm> #include<cmath> #include<cs ...

  2. POJ 1655.Balancing Act 树形dp 树的重心

    Balancing Act Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 14550   Accepted: 6173 De ...

  3. POJ3107Godfather[树形DP 树的重心]

    Godfather Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 6121   Accepted: 2164 Descrip ...

  4. POJ 2378.Tree Cutting 树形dp 树的重心

    Tree Cutting Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4834   Accepted: 2958 Desc ...

  5. 树形dp&&树的重心(D - Godfather POJ - 3107)

    题目链接:https://cn.vjudge.net/contest/277955#problem/D 题目大意:求树的重心(树的重心指的是树上的某一个点,删掉之后形成的多棵树中节点数最大值最小). ...

  6. hdu-4118 Holiday's Accommodation(树形dp+树的重心)

    题目链接: Holiday's Accommodation Time Limit: 8000/4000 MS (Java/Others)     Memory Limit: 200000/200000 ...

  7. 树形DP+树状数组 HDU 5877 Weak Pair

    //树形DP+树状数组 HDU 5877 Weak Pair // 思路:用树状数组每次加k/a[i],每个节点ans+=Sum(a[i]) 表示每次加大于等于a[i]的值 // 这道题要离散化 #i ...

  8. [HDU 5293]Tree chain problem(树形dp+树链剖分)

    [HDU 5293]Tree chain problem(树形dp+树链剖分) 题面 在一棵树中,给出若干条链和链的权值,求选取不相交的链使得权值和最大. 分析 考虑树形dp,dp[x]表示以x为子树 ...

  9. [poj3107]Godfather_树形dp_树的重心

    Godfather poj-3107 题目大意:求树的重心裸题. 注释:n<=50000. 想法:我们尝试用树形dp求树的重心,关于树的重心的定义在题目中给的很明确.关于这道题,我们邻接矩阵存不 ...

随机推荐

  1. for_each用法

    for_each()是个function template #include <algorithm>头文件说明 template<class _InIt, class _Fn1> ...

  2. Linux:查看磁盘空间占用情况

    Linux:查看磁盘空间占用情况 工作中有时被分配的测试机空间不大,经常遇到磁盘空间占满的情况.排查过程如下: 一.首先使用df -h 命令查看磁盘剩余空间,通过以下图看出/目录下的磁盘空间已经被占满 ...

  3. Arrays.asList()与toArray()

    Arrays.asList() 使用Arrays.asList()把数组转换成集合时,不能使用用于修改集合的方法(例如add.remove.clear),这将导致跑出UnsupportOperatio ...

  4. RegExp方法解析

    RegExp 正则表达式(regular expression)的缩写 语法: 显式构造函数 var rg= new RegExp(pattern, attributes) 隐式构造函数 var rg ...

  5. 20145222黄亚奇《网络对抗》 逆向及BOF进阶实践学习总结

    20145222<网络对抗> 逆向及BOF进阶实践学习总结 实践目的 1.注入shellcode 2.实现Return-to-libc攻击 知识点学习总结 Shellcode实际是一段代码 ...

  6. Cocos2d-x项目移植到WP8系列之九:使用自定义shader

    本文原链接:http://www.cnblogs.com/zouzf/p/3995132.html 有时候想得到一些例如灰度图等特殊的渲染效果,就得用到自定义shader,关于shader的一些背景知 ...

  7. mysql查询结果乱码

    1.查询数据库结果乱码 mysql,; +-----------+-------------------------------------------------------------+ | do ...

  8. Makefile文件应用——complicated项目

    学习资料 本文主要参考资料:驾驭Makefile(李云).pdf Complicated项目 需求: (1)object文件放到objs目录下 (2)可执行文件放到exes目录下 (3)增加头文件依赖 ...

  9. bootstrap重置校验方法

    $(function (){ $("select").change(function(){ $('#DepartForm').bootstrapValidator('resetFo ...

  10. javascript日期格式处理

    一. 服务端返回的日期和时间之间有T Asp.net MVC中 action返回前台的日期类型数据 是带有 T的,如: 2015-07-07T10:15:01. 这样的数据在Chrome浏览器,会自动 ...