码迷,mamicode.com
首页 > 其他好文 > 详细

HDU 6141 I am your Father!(最小树形图)

时间:2017-08-21 14:00:39      阅读:189      评论:0      收藏:0      [点我收藏+]

标签:roo   ++   str   tar   add   targe   答案   ret   std   

 

【题目链接】 http://acm.hdu.edu.cn/showproblem.php?pid=6141

 

【题目大意】

  给出一个有向图,求1点为根的最小树形图使得第n个点的直接父亲编号最小

 

【题解】

  如果没有第n个点直接父亲编号最小的要求,
  那么只要跑一遍朱刘算法即可,考虑到直接父亲最小的条件,
  我们连向第n个点的所有边进行加权操作,
  使得其在总边权相同的情况选取答案具有优先性

 

【代码】

#include <cstdio>
#include <algorithm>
using namespace std;
typedef long long LL; 
const int N=1010,M=10010;
const LL INF=0x3f3f3f3f3f3f3f3f;
int ROOT;
struct DMST{
    int n,size,pre[N],id[N],vis[N];
    LL in[N];
    struct EDGE{
        int u,v; LL cost;
        EDGE(){}
        EDGE(int a,int b,int c):u(a),v(b),cost(c){}
    }E[M];
    void init(int _n){n=_n,size=0;}
    void add(int u,int v,int w){E[size++]=EDGE(u,v,w);}
    LL dmst(int root){
        int u,v,cnt;
        LL ret=0;
        while(1){
            for(int i=0;i<n;i++)in[i]=INF;
            for(int i=0;i<size;i++){
                u=E[i].u,v=E[i].v;
                if(E[i].cost<in[v]&&u!=v){
                    pre[v]=u,in[v]=E[i].cost;
                    if(u==root)ROOT=i;
                }
            }
            for(int i=0;i<n;i++)if(i!=root&&in[i]==INF)return -1;
            cnt=in[root]=0;
            for(int i=0;i<n;i++)id[i]=vis[i]=-1;
            for(int i=0;i<n;i++){
                ret+=in[i],v=i;
                while(vis[v]!=i&&id[v]==-1&&v!=root)vis[v]=i,v=pre[v];
                if(v!=root&&id[v]==-1){
                    for(u=pre[v];u!=v;u=pre[u])id[u]=cnt;
                    id[v]=cnt++;
                }
            }
            if(!cnt)break;
            for(int i=0;i<n;i++)if(id[i]==-1)id[i]=cnt++;
            for(int i=0;v=E[i].v,i<size;i++){
                E[i].u=id[E[i].u],E[i].v=id[E[i].v];
                if(E[i].u!=E[i].v)E[i].cost-=in[v];
            }n=cnt,root=id[root];
        }return ret;
    }
}U;
int T,n,m;
int main(){
    scanf("%d",&T);
    while(T--){
        scanf("%d%d",&n,&m);
        U.init(n);
        while(m--){
            int x,y,z;
            scanf("%d%d%d",&x,&y,&z);
            z*=1000; if(y==n)z+=n-x;
            U.add(x-1,y-1,-z);
        }LL ans=-U.dmst(0);
        int fa=n-ans%1000;
        ans/=1000;
        printf("%lld %d\n",ans,fa);
    }return 0;
}

HDU 6141 I am your Father!(最小树形图)

标签:roo   ++   str   tar   add   targe   答案   ret   std   

原文地址:http://www.cnblogs.com/forever97/p/hdu6141.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!