zoukankan      html  css  js  c++  java
  • hdu 2196 Computer(树形DP经典)

    Computer

    Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
    Total Submission(s): 5925    Accepted Submission(s): 2979


    Problem Description
    A school bought the first computer some time ago(so this computer's id is 1). During the recent years the school bought N-1 new computers. Each new computer was connected to one of settled earlier. Managers of school are anxious about slow functioning of the net and want to know the maximum distance Si for which i-th computer needs to send signal (i.e. length of cable to the most distant computer). You need to provide this information.


    Hint: the example input is corresponding to this graph. And from the graph, you can see that the computer 4 is farthest one from 1, so S1 = 3. Computer 4 and 5 are the farthest ones from 2, so S2 = 2. Computer 5 is the farthest one from 3, so S3 = 3. we also get S4 = 4, S5 = 4.
     
    Input
    Input file contains multiple test cases.In each case there is natural number N (N<=10000) in the first line, followed by (N-1) lines with descriptions of computers. i-th line contains two natural numbers - number of computer, to which i-th computer is connected and length of cable used for connection. Total length of cable does not exceed 10^9. Numbers in lines of input are separated by a space.
     
    Output
    For each case output N lines. i-th line must contain number Si for i-th computer (1<=i<=N).
     
    Sample Input
    5 1 1 2 1 3 1 1 1
     
    Sample Output
    3 2 3 4 4
     
    Author
    scnu
     
    Recommend
    lcy
     
    #include<stdio.h>
    #include<iostream>
    #include<algorithm>
    #include<string.h>
    #include<vector>
    #define N 10010
    using namespace std;
    struct node
    {
        int to,len;//下一个节点,长度
        node (int x,int y)
        {
            to=x;
            len=y;
        }
    };
    int n;
    /*
    第一个dfs是先搜然后再转移状态的,用结点下方的数据更新
    第二个dfs是先状态转移然后再搜的,这是从结点上方进行数据更新,刚好满足题意:结点左右两个“子树”
    */
    vector<node> adj[N*2];
    int maxn[N];//第i个点的最大权值
    int smaxn[N];//第i个点的第二大的权值
    int maxi[N];//最大权值的点
    int smaxi[N];//第二大权值的点
    void dfs1(int u,int p)//p是u的父节点
    {
        maxn[u]=0;
        smaxn[u]=0;
        for(int i=0;i<adj[u].size();i++)
        {
            int v=adj[u][i].to;
            if(v==p) continue;//和父节点一样
            dfs1(v,u);
            if(maxn[v]+adj[u][i].len>smaxn[u])//先和第二长的距离比较,这样能记录下第二长的距离,这个数据能为下一个dfs提供判断
            {
                smaxn[u]=maxn[v]+adj[u][i].len;
                smaxi[u]=v;
                if(maxn[u]<smaxn[u])//更新之后第二长的路径,如果比原来最长的路径还长就替换,并且原来的最长的就变成第二长得了
                {
                    swap(maxn[u],smaxn[u]);
                    swap(maxi[u],smaxi[u]);
                }
            }
        }
    }
    
    //从父节点更新过来的
    void dfs2(int u,int p)
    {
        for(int i=0;i<adj[u].size();i++)
        {
            int v=adj[u][i].to;
            if(v==p) continue;//和父节点一样
            if(v==maxi[u])
            //这个地方如果下一个结点刚巧是就是dfs1中搜出来的最大值的节点的话
            //那么这一条路就不能用最大值去算,因为这样就算上上一条路的了
            {
                if(adj[u][i].len+smaxn[u]>smaxn[v])
                {
                    smaxn[v]=adj[u][i].len+smaxn[u];
                    smaxi[v]=u;
                    if(maxn[v]<smaxn[v])
                    {
                        swap(maxn[v],smaxn[v]);
                        swap(maxi[v],smaxi[v]);
                    }
                }
            }
            else
            {
                if(adj[u][i].len+maxn[u]>smaxn[v])
                {
                    smaxn[v]=adj[u][i].len+maxn[u];
                    smaxi[v]=u;
                    if(maxn[v]<smaxn[v])
                    {
                        swap(maxn[v],smaxn[v]);
                        swap(maxi[v],smaxi[v]);
                    }
                }
            }
            dfs2(v,u);
        }
    }
    int main()
    {
        //freopen("in.txt","r",stdin);
        while(scanf("%d",&n)!=EOF)
        {
            for(int i=0;i<=n;i++)
                adj[i].clear();
            for(int i=2;i<=n;i++)
            {
                int a,b;
                scanf("%d%d",&a,&b);
                adj[i].push_back(node(a,b));
                adj[a].push_back(node(i,b));
            }
            dfs1(1,-1);//从子节点更新
            dfs2(1,-1);//从父节点更新
            for(int i=1;i<=n;i++)
                printf("%d
    ",maxn[i]);
        }
        return 0;
    }
  • 相关阅读:
    001 :PCL 的基本文件类型PCD的读入和写入操作
    CMake +Vs2017+快速搭建pcl1.9.1环境
    Window 10 PCL-1.91+VS2017 源码编译以及安装pcl
    Eigen3+Cmake+Vs2017源码编译
    将Opencv加入到环境变量中
    004 :opencv 中矩阵操作以及通过内存的方式取像素
    ubuntu16.04与win10双系统安装 无法将grub-efi-amd64-signed 软件包安装到/target/中
    简单了解一下PyTest-sq
    软件测试工程师笔试题
    TT-反射-对象拷贝
  • 原文地址:https://www.cnblogs.com/wuwangchuxin0924/p/5794419.html
Copyright © 2011-2022 走看看