【P2052】道路修建(树形+搜索)

这个题看上去高大上,实际上就是一个大水题。怎么说呢,这个题思路可能比较难搞,代码实现难度几乎为0.

首先我们可以发现这是一棵树,然后问其中任意一条边左右两边的点的数量之差的绝对值,实际上,无论两边的点是多少,我们都可以用abs(n-2*x)(x代表这条边的一个端点的儿子的个数)算出来,这样我们就减少了一大部分我们的工作,我们只需要在dfs遍历树的时候在返回的时候进行一个子节点的增加和和的计算,这个题就可以轻松得出结果。讨论里说要开ll,所以蒟蒻一遍A了。。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define re register
using namespace std;
struct po
{
    int next;
    int to;
    int dis;
};
po edge[1000001];
long long head[1000001],b[1000001],temp[1000001],tot[1000001],ans;
long long dis[1000001];
int T,s,t,n,m,x,w,a,l,num,flag,d;
inline void add_edge(int from,int to,int dis)
{
    edge[++num].next=head[from];
    edge[num].to=to;
    edge[num].dis=dis;
    head[from]=num;
}
inline void dfs(int to,int fa)
{
    tot[to]=1;
    for(re int i=head[to];i;i=edge[i].next)
    {
        if(edge[i].to==fa) continue;
        dfs(edge[i].to,to);
        tot[to]+=tot[edge[i].to];
        ans+=edge[i].dis*abs(n-2*tot[edge[i].to]);
    }
}
int main()
{
    cin>>n;
    for(re int i=1;i<=n-1;i++)
    {
        cin>>s>>t>>d;
        add_edge(s,t,d);
        add_edge(t,s,d);
    }
    dfs(1,0);
    cout<<ans;
}

 

posted @ 2018-02-07 15:51  ~victorique~  阅读(175)  评论(0编辑  收藏  举报
Live2D