题目大意:给定一个有向图,每个点出度为1,让你将这些点分为两个集合x和y,使得对于每一个y集合内的点都存在一个x集合内的点指向它,并且使得y集合尽可能大。
原图给的是一些内向基环树,显然要求的是一个最小支配集@poj3659。内向的基环树我们不好处理,考虑将这颗树反向,变成了外向基环树,这样我们可以随便找到环上的一个点以及指向它的点(首先每个联通快一定有环,但其实没有环的话就是正常的树了),不妨设为y->x,枚举x选择还是不选择,如果x选了,那么y就有人支配了,以x为根做正常的最小支配集即可。
做最小值配集的时候有一个小技巧tmp,有兴趣可以阅读下…
(还有一种直接贪心乱搞的做法在下面…)
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
//by:MirrorGray
using namespace std;
const int N=1111111;
int tot=-1,head[N],ver[N],nxt[N];
int root,f[N],g[N],fa[N],vis[N],ban[N];
void add(int x,int y){
nxt[++tot]=head[x];
head[x]=tot;
ver[tot]=y;
}
int find(int x){
vis[x]=true;
return vis[fa[x]]?x:find(fa[x]);
}
void dfs(int x){
vis[x]=true;f[x]=1;g[x]=0;