zoukankan      html  css  js  c++  java
  • hiho 1564

    题目链接

    H公司有 N 台服务器,编号1~N,组成了一个树形结构。其中中央服务器处于根节点,终端服务器处于叶子节点。  

    中央服务器会向终端服务器发送消息。一条消息会通过中间节点,到达所有的终端服务器。消息每经过一台服务器(包括根节点的中央服务器和叶子节点的终端服务器),就会被处理并存储下来,其中编号为 i 的服务器,执行处理和存储这个过程会花费 Ai 毫秒的时间。消息从一台服务器传输到另一台服务器的时间可以忽略不计。  

    由于 Ai 各不相同,从中央服务器发出消息到各个终端处理并储存完毕的时间也是不相同的。例如如下的网络,假设1是中央服务器,而2和4是终端服务器。

       1
      / 
     2   3
         |
         4

    如果A1=1, A2=4, A3=3, A4=2,那么假设1发出消息的时刻是0,终端2处理完的时刻是A1+A2=5,而终端4处理完的时刻是A1+A3+A4=6。  

    CEO交给小Ho一个任务,要求每台终端处理完消息的时间是完全一致的。小Ho想到了一个天才的解决方案:强行增加某些服务器的处理时间 Ai。例如在上例中,将 A2从4增加到5即可使所有终端都在6毫秒后处理完毕。

    当然小Ho希望所有服务器增加的处理时间总和最少。你能帮助小Ho吗?

    ------------------------------------------------------------------------------------------------------------------------------------

    一次遍历,记录每个节点到叶节点的最长路径的长度,以此为依据进行调整。

    wa了好几次才找到原因:

    //maxc = MAX(maxc,dfs(to));
    maxc = std::max(maxc,dfs(to));

    开始用的宏定义MAX

    #define MAX(a,b) ((a)>=(b)?(a):(b))

    相当于调用了两次dfs()。。。。

    #include <set>
    #include <map>
    #include <stack>
    #include <queue>
    #include <cmath>
    #include <vector>
    #include <string>
    #include <cstdio>
    #include <cstring>
    #include <cstdlib>
    #include <iostream>
    #include <algorithm>
    
    #define MAX(a,b) ((a)>=(b)?(a):(b))
    #define MIN(a,b) ((a)<=(b)?(a):(b))
    #define OO 0x0fffffff
    using namespace std;
    typedef long long LL;
    const int N = 1E5+7;
    struct Edge{
        int to;
        int next;
    };
    int cost[N];
    LL maxCost[N];
    int heads[N];
    Edge edgs[N];
    bool degree[N];
    int eid = 0;
    LL ans = 0;
    LL dfs(int id){
        LL maxc = 0;
        for(int cid = heads[id];cid!=-1;cid=edgs[cid].next){
            int to = edgs[cid].to;
            //maxc = MAX(maxc,dfs(to));
            maxc = std::max(maxc,dfs(to));
        }
        for(int cid = heads[id];cid!=-1;cid=edgs[cid].next){
            int to = edgs[cid].to;
            ans += maxc - maxCost[to];
        }
        return maxCost[id] = maxc+cost[id];
    }
    
    int main(){
        int n,a,b;
        while(scanf("%d",&n)!=EOF){
            eid = 0;
            memset(heads,-1,sizeof(heads));
            memset(degree,false,sizeof(degree));
            for(int i=0;i<n;i++) scanf("%d",cost+i);
            for(int i=0;i<n-1;i++){
               scanf("%d%d",&a,&b); a--,b--;
               edgs[eid].to = b;
               edgs[eid].next = heads[a];
               heads[a] = eid++;
               degree[b] = true;
            }
            int root; for(int i=0;i<n;i++) if(!degree[i]) {root=i;break;}
            ans = 0;
            dfs(root);
            printf("%lld
    ",ans);
        }
        return 0;
    }
  • 相关阅读:
    css 两边是线,中间文字的多种实现方法
    vue provide/inject 父组件如何给孙子组件传值
    Mac版本的 Axure rp8 不显示菜单栏
    mac 如何卸载node和npm采坑之旅
    css3 鼠标悬停图片动画
    css3 一个六边形 和 放大旋转动画DEMO演示
    js drag drop 收藏夹拖拽移除的简单例子
    css 折角效果/切角效果
    css 给图片添加滤镜效果,透明层毛玻璃效果
    c# udp通讯实现
  • 原文地址:https://www.cnblogs.com/redips-l/p/7463380.html
Copyright © 2011-2022 走看看