【题解】codeforces 219D Choosing Capital for Treeland 树型dp
题目描述
Treeland国有n个城市,这n个城市连成了一颗树,有n-1条道路连接了所有城市。每条道路只能单向通行。现在政府需要决定选择哪个城市为首都。假如城市i成为了首都,那么为了使首都能到达任意一个城市,不得不将一些道路翻转方向,记翻转道路的条数为k。你的任务是找到所有满足k最小的首都。
输入输出格式
输入格式
输入包含多个测试点。对于每个测试点,每个测试点的第一行为一个正整数n(2<=n<=2e5)。接下来n-1行,每行两个正整数ai,bi,表示城市a到城市b有一条单向通行的道路。输入以空格分隔,以EOF结尾。
输出格式
对于每个测试点,第一行输出k,第二行升序输出所有满足条件的首都的编号。
题目描述
The country Treeland consists of n cities, some pairs of them are connected with unidirectional roads. Overall there are n−1 roads in the country. We know that if we don’t take the direction of the roads into consideration, we can get from any city to any other one.
The council of the elders has recently decided to choose the capital of Treeland. Of course it should be a city of this country. The council is supposed to meet in the capital and regularly move from the capital to other cities (at this stage nobody is thinking about getting back to the capital from these cities). For that reason if city a is chosen a capital, then all roads must be oriented so that if we move along them, we can get from city a to any other city. For that some roads may have to be inversed.
Help the elders to choose the capital so that they have to inverse the minimum number of roads in the country.
输入输出格式
输入格式:
The first input line contains integer $n$( $2<=n<=2·10^{5}$ ) — the number of cities in Treeland. Next n−1 n-1 n−1 lines contain the descriptions of the roads, one road per line. A road is described by a pair of integers $s_{i}$,$t_{i}$ ( $1<=s_{i}$,$t_{i}<=n$; $s_{i}≠t_{i}$ ) — the numbers of cities, connected by that road. The $i$-th road is oriented from city $s_{i}$ to city $t_{i}$ . You can consider cities in Treeland indexed from 1 to $n$ .
输出格式:
In the first line print the minimum number of roads to be inversed if the capital is chosen optimally. In the second line print all possible ways to choose the capital — a sequence of indexes of cities in the increasing order.
输入输出样例
输入样例#1: 复制
3
2 1
2 3
输出样例#1: 复制
0
2
输入样例#2: 复制
4
1 4
2 4
3 4
输出样例#2: 复制
2
1 2 3
思路
树型dp+两遍dfs
- 在第一次自底向上的dfs中,我们用$dp[u]$表示以u节点为首都时,u到其子树所有节点S需要逆转的边数
- 令正向边权值为0,反向边权值为1 得转移方程 $dp[u]=∑(dp[s]+w[i,s])$
- 第二次 $dp[i]$的定义变成了u到全树节点需要逆转的边数
分情况讨论
设 f为u的父亲
- 当f->u这条边是正向的时候,以u为首都则需要将这条边逆转, $dp[u]+=dp[f]+1$
- 当f->u这条边是反向的时候,以u为首都不需要将这条边逆转,但由于u是f的子节点,因此在dp[f]中将这条边逆转了,所以 $dp[u]+=dp[f]-1$
代码
#include<cmath>
#include<cstdio>
#include<string>
#include<cstring>
#include<iostream>
#include<algorithm>
#define re register int
using namespace std;
inline int read(){
int x=0,w=1;
char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
if(ch=='-') w=-1,ch=getchar();
while(ch>='0'&&ch<='9') x=(x<<1)+(x<<3)+ch-48,ch=getchar();
return x*w;
}
struct edge {
int to,next;
bool w;
}e[400010];
int head[200010],tot,dp[200010];
inline void add(int from,int to,bool w) {
e[++tot]=(edge){to,head[from],w};
head[from]=tot;
}
void dfs1(int u,int fa) { //求u到子节点需要逆转的边数
for(int i=head[u];i;i=e[i].next) {
int v=e[i].to;
if(v==fa) continue;
dfs1(v,u);
dp[u]+=dp[v]+e[i].w;
}
}
void dfs2(int u,int fa) { //求u到全树需要逆转的边数
for (int i=head[u];i;i=e[i].next) {
int v=e[i].to;
if (v==fa) continue;
dp[v]=dp[u]+(e[i].w?-1:1);
dfs2(v,u);
}
}
int main() {
int n,a,b;
while (scanf("%d",&n)!=EOF) {
tot=0;
memset(head,0,sizeof(head));
for (int i=1;i<=n-1;i++) {
a=read(),b=read();
add(a,b,0);
add(b,a,1);
}
memset(dp,0,sizeof(dp));
dfs1(1,-1);dfs2(1,-1);
int Min=99999999;
for (int i=1;i<=n;i++) if (Min>dp[i]) Min=dp[i];
printf("%d\n",Min);
for (int i=1;i<=n;i++) if (Min==dp[i]) printf("%d ",i);
printf("\n");
}
return 0;
}
【题解】codeforces 219D Choosing Capital for Treeland 树型dp的更多相关文章
- Codeforces 219D. Choosing Capital for Treeland (树dp)
题目链接:http://codeforces.com/contest/219/problem/D 树dp //#pragma comment(linker, "/STACK:10240000 ...
- Codeforces 219D Choosing Capital for Treeland:Tree dp
题目链接:http://codeforces.com/problemset/problem/219/D 题意: 给你一棵树,n个节点. 树上的边都是有向边,并且不一定是从父亲指向儿子的. 你可以任意翻 ...
- CodeForces 219D Choosing Capital for Treeland (树形DP)
题意:给一个树形图,n个节点,n-1条有向边,要求选一个节点作为根,使需要改变方向的边的数目最少.并输出所有可能作为根的点. 思路: 先随便一个点进行DFS,计算将每棵子树的边全部往下时,所需要的费用 ...
- Codeforces 219D Choosing Capital for Treeland(树形DP)
题目是给一张边有向的树形图.要选出首都的点,首都要都能走到其他点,因此要反转一些边的方向.问可以选哪几个点作为首都,使它们所需反转边的数量最少. 这题挺好想的,因为做过HDU2196. 首先就不妨设正 ...
- Codeforces 219D Choosing Capital for Treeland 2次DP
//选择一个根使得变换最少边的方向使得能够到达所有点#include <map> #include <set> #include <list> #include & ...
- (纪念第一道完全自己想的树DP)CodeForces 219D Choosing Capital for Treeland
Choosing Capital for Treeland time limit per test 3 seconds memory limit per test 256 megabytes inpu ...
- Codeforces 219D Choosing Capital for Treeland
http://codeforces.com/problemset/problem/219/D 题目大意: 给出一棵树,但是它的边是有向边,选择一个城市,问最少调整多少条边的方向能使一个选中城市可以到达 ...
- Codeforces 219D - Choosing Capital for Treeland(树形dp)
http://codeforces.com/problemset/problem/219/D 题意 给一颗树但边是单向边,求至少旋转多少条单向边的方向,可以使得树上有一点可以到达树上任意一点,若有多个 ...
- CodeForces 219D Choosing Capital for Treeland (树形DP)经典
<题目链接> 题目大意: 给定一个有向树,现在要你从这颗树上选一个点,使得从这个点出发,到达树上其它所有点所需翻转的边数最小,输出最少需要翻转的边数,并且将这些符合条件的点输出. 解题分析 ...
随机推荐
- CentOS7 搭建 Redis 集群
一.手动搭建 1. 准备节点 节点数量至少为 6 个才能保证组成完整高可用的集群 (1) 目录结构 cluster ├── 9001 │ ├── data │ │ ├── appendon ...
- 段间跳转之TSS段
TR寄存器,TSS描述符,TSS段 TR寄存器与普通的段寄存器一样都有可见部分和不可见部分.TR的可见部分为16位为其段选择子,不可见部分是32位的TSS基地址和16位的大小. TSS描述符存在GDT ...
- volatile关键字的作用-respect
volatile关键字的含义? volatile定义的变量可能会意外的改变,改变它的情况有很多(例如:操作系统,硬件,线程),编译就不会去假设这个值,也就是说每次访问这个变量时,系统就会小心翼翼的去从 ...
- ES6学习-1 啥是ES6啊
我不准备刨根问底,找寻他的十八辈祖先,希望用最简单的方式能够说明白ES6是个啥,come on baby ES6的全称是ECMAScript 6.0,他于2015 年 6 月正式发布,它的目标是使得 ...
- python基础之模块初识
Python的强大之处在于他有非常丰富和强大的标准库和第三方库,几乎你想实现的任何功能都有相应的Python库支持 一.time模块和datetime模块 和时间有关系的我们就要用到时间模块.在使用模 ...
- Mac 使用 Parallels Desktop 虚拟机安装 win10 教程
Parallels Desktop 介绍 Parallels Desktop 是一款运行在 Mac 电脑上的极为优秀的虚拟机软件,用户可以在 Mac OS X下非常方便运行 Windows.Linux ...
- Linux Access.conf安全配置
access.conf is the configuration file used to logins to the Linux or Unix systems. This file is loca ...
- 完全理解Python 迭代对象、迭代器、生成器
在了解Python的数据结构时,容器(container).可迭代对象(iterable).迭代器(iterator).生成器(generator).列表/集合/字典推导式(list,set,dict ...
- 结合JVM 浅谈Java 类加载器(Day_03)
所谓错过,不是错了,而是过了. 什么是JAVA类加载? Class对象由JVM自动产生,每当一个类被加载时,JVM就自动为其生成一个Class对象,通过Class对象可以获得类的相关信息.将类信息读取 ...
- Python break/continue - Python零基础入门教程
目录 一.break 二.continue 三.重点总结 四.猜你喜欢 零基础 Python 学习路线推荐 : Python 学习目录 >> Python 基础入门 在 Python wh ...