POJ 1655 Balancing Act && POJ 3107 Godfather
题目大意:
根据题目的图很好理解意思,就是记录每一个点的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的更多相关文章
- POJ.1655 Balancing Act POJ.3107 Godfather(树的重心)
关于树的重心:百度百科 有关博客:http://blog.csdn.net/acdreamers/article/details/16905653 1.Balancing Act To POJ.165 ...
- POJ 1655 Balancing Act&&POJ 3107 Godfather(树的重心)
树的重心的定义是: 一个点的所有子树中节点数最大的子树节点数最小. 这句话可能说起来比较绕,但是其实想想他的字面意思也就是找到最平衡的那个点. POJ 1655 题目大意: 直接给你一棵树,让你求树的 ...
- poj 1655 Balancing Act 求树的重心【树形dp】
poj 1655 Balancing Act 题意:求树的重心且编号数最小 一棵树的重心是指一个结点u,去掉它后剩下的子树结点数最少. (图片来源: PatrickZhou 感谢博主) 看上面的图就好 ...
- poj 1655 Balancing Act(找树的重心)
Balancing Act POJ - 1655 题意:给定一棵树,求树的重心的编号以及重心删除后得到的最大子树的节点个数size,如果size相同就选取编号最小的. /* 找树的重心可以用树形dp或 ...
- POJ 1655 Balancing Act【树的重心】
Balancing Act Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14251 Accepted: 6027 De ...
- POJ 1655.Balancing Act 树形dp 树的重心
Balancing Act Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14550 Accepted: 6173 De ...
- POJ 1655 - Balancing Act 树型DP
这题和POJ 3107 - Godfather异曲同工...http://blog.csdn.net/kk303/article/details/9387251 Program: #include&l ...
- POJ 1655 Balancing Act 树的重心
Balancing Act Description Consider a tree T with N (1 <= N <= 20,000) nodes numbered 1...N. ...
- POJ 1655 - Balancing Act - [DFS][树的重心]
链接:http://poj.org/problem?id=1655 Time Limit: 1000MS Memory Limit: 65536K Description Consider a tre ...
随机推荐
- codechef: ADAROKS2 ,Ada Rooks 2
又是道原题... (HDU 6313 Hack It , 多校 ACM 里面的题) 题目说构造一个 n * n 矩阵,染色点不得构成矩形...然后染色点个数至少 8 * n 然后我们生成一个数 m , ...
- mysql left join 出现的结果会重复
left join 基本用法 MySQL left join 语句格式 A LEFT JOIN B ON 条件表达式 left join 是以A表为基础,A表即左表,B表即右表. 左表(A)的记录会全 ...
- JMeter配置MongoDB
1.启动JMeter,右键添加->配置文件->MongoDB Source Config. 注意:JMeter 3.0以上版本去掉了此配置项,可以从低版本处拷贝. 2.设置MongoDB配 ...
- Docker学习系列(二):Docker三十分钟快速入门(上)
一.背景 最近,Docker技术真是一片火热,它的出现也弥补了虚拟机资源消耗过高的问题,直接让虚拟化技术有了质的飞跃.那么本文我们来聊一聊Docker,和大家一起认识Docker,简单入门Dock ...
- [转]MVC4项目中验证用户登录一个特性就搞定
本文转自:http://www.mrhuo.com/Article/Details/470/A-Attribute-For-MVC4-Project-Used-To-Validate-User-Log ...
- C#基础 函数部分
函数:能够独立完成某项功能的模块. 函数四要素:输入.输出.函数体.函数名 函数定义: (static/public) 返回类型 函数名(参数类型 参数名,参数类型 参数名){ 函数体} 函数的调用: ...
- C#手机充值系统开发(基于聚合数据)
说是手机充值系统有点装了,其实就是调用了聚合数据的支付接口,其实挺简单的事 但是我发现博客园竟然没有类似文章,我就个出头鸟把我的代码贡献出来吧 首先说准备工作: 去聚合数据申请账号-添加手机支付的认证 ...
- 【JAVA 学习笔记2】if使用例子
int a =3; if (a%2==0) { System.out.println(a+" 是偶数"); System.out.println(a+" 不是奇数&quo ...
- IFormattable,ICustomFormatter, IFormatProvider接口
定 义 1.IFormattable 提供一种功能,用以将对象的值格式化为字符串表示形式. 2.IFormatProvider 提供用于检索控制格式化的对象的机制 ...
- Android中出现Error:In (declare-styleable) FontFamilyFont, unable to find attribute android:font
Android中出现Error:In (declare-styleable) FontFamilyFont, unable to find attribute android:font 解决办法,今天 ...