给一棵树 每次可以选一个度数为偶数的点删掉。问能不能删完所有点。如果能 输出任意方案。
我们注意到 给的树如果是偶数点 就会有奇数条边。然而我们每个操作一定删掉偶数条边。所以一棵偶数点的树是不能被删完的。
如果有奇数点的树。删掉任意一点后,他的子树就会变成森林。如果有偶数点的树是删不掉的。
因此,对于一个点他的子树的
size
s
i
z
e
是奇数的,那么可以先删掉父亲再删掉子树,否则先删掉子树的根再删掉父亲。
这样关系就形成了一个有向无环图。对
DAG
D
A
G
拓扑排序即可。
#include<bits/stdc++.h>
using namespace std;
const int MAXN=5e5+5;
struct edge{
int to,next;
}e[MAXN<<1],dag[MAXN<<1];
int head[MAXN],cnt=0,n,headdag[MAXN],cntdag=0,in[MAXN];
inline void add(int u,int v){
e[++cnt]=(edge){v,head[u]},head[u]=cnt;
e[++cnt]=(edge){u,head[v]},head[v]=cnt;
}
inline void adddag(int u,int v){
dag[++cntdag]=(edge){v,headdag[u]},headdag[u]=cntdag,in[v]++;
}
int size[MAXN];
void dfs(int u,int fa){
size[u]=1;
for(int i=head[u];i;i=e[i].next){
int v=e[i].to;
if(v==fa)continue;
dfs(v,u);
size[u]+=size[v];
}
}
void getin(int u,int fa){
for(int i=head[u];i;i=e[i].next){
int v=e[i].to;
if(v==fa)continue;
if(size[v]&1)adddag(u,v);
else adddag(v,u);
getin(v,u);
}
}
queue<int>q;
void getans(){
puts("YES");
for(int i=1;i<=n;i++)if(!in[i])q.push(i);
while(q.size()){
int u=q.front();q.pop();
printf("%d\n",u);
for(int i=headdag[u];i;i=dag[i].next){
int v=dag[i].to;
in[v]--;
if(!in[v])q.push(v);
}
}
}
int main(){
scanf("%d",&n);
if(!(n&1)){
puts("NO");
return 0;
}
for(int i=1;i<=n;i++){
int tmp;
scanf("%d",&tmp);
if(tmp)add(tmp,i);
}
dfs(1,1);
getin(1,1);
getans();
return 0;
}