Codeforce 219D(树形dp)

本文探讨了一种通过反转有向树中部分边的方式,以确保任意节点都能到达选定的首都节点,并且达到所有节点的路径和最小化的算法。详细介绍了DFS遍历过程以及如何通过递归调用实现路径和的更新和最小路径的查找。

链接:点击打开链接

题意:由n各节点组成的有向树,要求选一个点作为首都,通过反转一些边使得任何一个点都能到达首都,输出反转的最少次数和可能成为首都的节点编号

代码:

#include <vector>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <iostream>
#include <algorithm>
using namespace std;
const int SIZE=200005;
const int INF=0x3f3f3f3f;
int dp[SIZE],num[SIZE],used[SIZE];
struct node{
    int to,cost;
};
vector<node> G[SIZE];
void dfs1(int s,int sum){
    int i,tmp;
    dp[1]+=sum;
    used[s]=1;
    for(i=0;i<G[s].size();i++){
        tmp=G[s][i].to;
        if(G[tmp].size()&&!used[tmp]){
        dfs1(tmp,G[s][i].cost);
        }
    }
}                                               //以1为根节点,通过根节点到其他节点的
void dfs2(int s){                               //路径和推出其他节点的路径和
    int i,tmp;
    used[s]=1;
    for(i=0;i<G[s].size();i++){
        tmp=G[s][i].to;
        if(!used[tmp]){
            if(G[s][i].cost==0)                 //根据父节点推出子节点
            dp[tmp]=dp[s]+1;
            else
            dp[tmp]=dp[s]-1;
            dfs2(tmp);
        }
    }
}
int main(){                                     //就是将正向边设为1,反向边设为0
    int n,i,j,a,b,ans;                          //求哪一个节点到其他的路径和最小
    while(scanf("%d",&n)!=EOF){
        ans=INF;
        for(i=1;i<=n;i++)
        G[i].clear();
        memset(dp,0,sizeof(dp));
        memset(used,0,sizeof(used));
        for(i=0;i<n-1;i++){
            scanf("%d%d",&a,&b);
            G[b].push_back((node){a,1});
            G[a].push_back((node){b,0});
        }                                       //建树
        dfs1(1,0);
        memset(used,0,sizeof(used));
        dfs2(1);
        for(i=1;i<=n;i++)
        ans=min(ans,dp[i]);                     //找出最小值
        printf("%d\n",ans);
        for(i=1;i<=n;i++)
        if(dp[i]==ans)
        printf("%d ",i);
        printf("\n");
    }
    return 0;
}


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值