题目大意:

根据题目的图很好理解意思,就是记录每一个点的balance,例如 i 的balance就是把 i 从这棵树中除去后得到的森林中含有结点数最多

的子树中的节点个数,然后找到所有节点中对应的balance的最小值 , 并输出最小值对应的最小的标号

题目不要看花。。。前一个是找最大,后面找所有最大值中的最小,我就是看错一直wa,后来人家题解刚看完题目意思就反应过来 , 囧死了 - -

昨天貌似做过一道类似的题,任意找一点 , 比如 1 作为根进行dfs,第一次dfs记录所有点对应的子树中含有的节点的总个数

第二次dfs记录down[i] , 也就是i子树中 i 连接的所有子树中含有节点数最多的节点个数

其实down[i]就相当 i 出去后 , i 下方森林可得到的最大balance , 还需要求一个 i 上方的balance , 这个上方的值就是n-sum[i]

两者比较取最大就可以了

 #include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
const int N = ;
int first[N] , k , sum[N] , down[N] , rec[N]; struct Edge{
int y , next;
}e[N<<]; void add_edge(int x , int y)
{
e[k].y = y , e[k].next = first[x];
first[x] = k++;
} void dfs1(int u , int fa)
{
sum[u] = ;
for(int i=first[u] ; i!=- ; i=e[i].next)
{
int v = e[i].y;
if(v == fa) continue;
dfs1(v , u);
sum[u] += sum[v];
}
}
void dfs2(int u , int fa)
{
for(int i=first[u] ; i!=- ; i=e[i].next)
{
int v = e[i].y;
if(v == fa) continue;
down[u] = max(sum[v]+ , down[u]);
dfs2(v , u);
}
} int main()
{
// freopen("a.in" , "r" , stdin);
int T;
scanf("%d" , &T);
while(T--)
{
int n , a , b;
scanf("%d" , &n);
memset(first , - , sizeof(first));
k = ;
for(int i= ; i<n ; i++){
scanf("%d%d" , &a , &b);
add_edge(a , b);
add_edge(b , a);
}
memset(down , , sizeof(down));
dfs1( , -);
dfs2( , -);
int minn = down[]-;
rec[] = down[]-;
// cout<<"down: 1: "<<down[1]<<" "<<sum[1]<<endl;
for(int i= ; i<=n ; i++){
// cout<<"down: i: "<<i<<" "<<" "<<down[i]<<" "<<sum[i]<<endl;
rec[i] = max(down[i]- , n-sum[i]);
minn = min(minn , rec[i]);
}
int i;
for(i= ; i<=n ; i++){
if(rec[i] == minn) break;
}
printf("%d %d\n" , i , minn);
}
return ;
}

POJ 3107

求解问题的思想是和上面的题目是基本相同的,除了输入输出方式需要改变,然后数组开大点就可以照抄上面代码了

 #include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
const int N = ;
int first[N] , k , sum[N] , down[N] , rec[N]; struct Edge{
int y , next;
}e[N<<]; void add_edge(int x , int y)
{
e[k].y = y , e[k].next = first[x];
first[x] = k++;
} void dfs1(int u , int fa)
{
sum[u] = ;
for(int i=first[u] ; i!=- ; i=e[i].next)
{
int v = e[i].y;
if(v == fa) continue;
dfs1(v , u);
sum[u] += sum[v];
}
}
void dfs2(int u , int fa)
{
for(int i=first[u] ; i!=- ; i=e[i].next)
{
int v = e[i].y;
if(v == fa) continue;
down[u] = max(sum[v]+ , down[u]);
dfs2(v , u);
}
} int main()
{
// freopen("a.in" , "r" , stdin);
int n , a , b;
while(scanf("%d" , &n) == ){
memset(first , - , sizeof(first));
k = ;
for(int i= ; i<n ; i++){
scanf("%d%d" , &a , &b);
add_edge(a , b);
add_edge(b , a);
}
memset(down , , sizeof(down));
dfs1( , -);
dfs2( , -);
int minn = down[]-;
rec[] = down[]-;
// cout<<"down: 1: "<<down[1]<<" "<<sum[1]<<endl;
for(int i= ; i<=n ; i++){
// cout<<"down: i: "<<i<<" "<<" "<<down[i]<<" "<<sum[i]<<endl;
rec[i] = max(down[i]- , n-sum[i]);
minn = min(minn , rec[i]);
}
int num = ;
for(int i= ; i<=n ; i++){
if(rec[i] == minn){
if(num == ) printf("%d" , i);
else printf(" %d" , i);
num++;
}
}
}
return ;
}

POJ 1655 Balancing Act && POJ 3107 Godfather的更多相关文章

  1. POJ.1655 Balancing Act POJ.3107 Godfather(树的重心)

    关于树的重心:百度百科 有关博客:http://blog.csdn.net/acdreamers/article/details/16905653 1.Balancing Act To POJ.165 ...

  2. POJ 1655 Balancing Act&&POJ 3107 Godfather(树的重心)

    树的重心的定义是: 一个点的所有子树中节点数最大的子树节点数最小. 这句话可能说起来比较绕,但是其实想想他的字面意思也就是找到最平衡的那个点. POJ 1655 题目大意: 直接给你一棵树,让你求树的 ...

  3. poj 1655 Balancing Act 求树的重心【树形dp】

    poj 1655 Balancing Act 题意:求树的重心且编号数最小 一棵树的重心是指一个结点u,去掉它后剩下的子树结点数最少. (图片来源: PatrickZhou 感谢博主) 看上面的图就好 ...

  4. poj 1655 Balancing Act(找树的重心)

    Balancing Act POJ - 1655 题意:给定一棵树,求树的重心的编号以及重心删除后得到的最大子树的节点个数size,如果size相同就选取编号最小的. /* 找树的重心可以用树形dp或 ...

  5. POJ 1655 Balancing Act【树的重心】

    Balancing Act Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 14251   Accepted: 6027 De ...

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

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

  7. POJ 1655 - Balancing Act 树型DP

    这题和POJ 3107 - Godfather异曲同工...http://blog.csdn.net/kk303/article/details/9387251 Program: #include&l ...

  8. POJ 1655 Balancing Act 树的重心

    Balancing Act   Description Consider a tree T with N (1 <= N <= 20,000) nodes numbered 1...N. ...

  9. POJ 1655 - Balancing Act - [DFS][树的重心]

    链接:http://poj.org/problem?id=1655 Time Limit: 1000MS Memory Limit: 65536K Description Consider a tre ...

随机推荐

  1. sql注入方法以及防范

    sql注入方法: 1.数字注入 ; get请求 www.bobo.com?id=1 可以查出 ID等于1的一条数据. 如果有人在链接后面增加  www.bobo.com?id=1 or 1=1 / w ...

  2. redis在linux的安装和开机启动(二)

    编译 安装 makefile已经存在 执行make 即可 make之后, 自动创建可运行的脚本文件, 不需要再执行 install了. 将脚本文件, 拷贝到指定位置, 就可以了. 手动创建目录, 需要 ...

  3. 字符类型C++(ascll码表)

    ascll码: 序号 字符 序号 字符 序号 字符 序号 字符 序号 字符 序号 字符 32 空格 48 0 64 @ 80 P 96 ` 112 p 33 ! 49 1 65 A 81 Q 97 a ...

  4. less新手入门(四)—— Mixin Guards

    八.Mixin Guards 有条件的 mixin 当您想要匹配表达式时,相对于简单的值或特性,Guards是有用的.如果您熟悉函数式编程,您可能已经遇到过它们. 为了尽可能地保持CSS的声明性质,在 ...

  5. 自动生成 html5 小页面

    StringBuilder htmltext = new StringBuilder();            try            {                //var readP ...

  6. 微信小程序授权 获取用户的openid和session_key【后端使用java语言编写】,我写的是get方式,目的是测试能否获取到微信服务器中的数据,后期我会写上post请求方式。

    在这里给大家分享下我的心得,1.写代码前一定要对整个流程有个了解.我就是因为在先不了解整个过程中去ctrl+c+v他人的博客代码,花费很多无用的时间去处理还不知道能不能跑的起来的代码. 2.本人比较喜 ...

  7. es6之iterator,for...of

    遍历器(Iterator)是一种统一的接口机制,来处理所有不同的数据结构. JavaScript 原有的表示“集合”的数据结构,主要是数组(Array)和对象(Object),ES6 又添加了Map和 ...

  8. Spark学习之编程进阶——累加器与广播(5)

    Spark学习之编程进阶--累加器与广播(5) 1. Spark中两种类型的共享变量:累加器(accumulator)与广播变量(broadcast variable).累加器对信息进行聚合,而广播变 ...

  9. Jmeter各组件介绍 及 使用

    本篇主要讲述Jmeter的各个组件及简单使用,其中包括以下内容: 一.线程组二.逻辑控制器三.配置元件四.定时器五.后置处理器六.断言七.监听器 八.参数化 网上大神整理的链接:http://blog ...

  10. Pro ASP.NET Core MVC 第6版 第二章(前半章)

    目录 第二章 第一个MVC 应用程序 学习一个软件开发框架的最好方法是跳进他的内部并使用它.在本章,你将用ASP.NET Core MVC创建一个简单的数据登录应用.我将它一步一步地展示,以便你能看清 ...