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

    求出每一点到所有点的最大距离,刚开始就有点想法求最长边的问题,但想了一会不知道怎么去把子节点转化成根节点。

    任选一点为根节点,先求出每个节点到子节点的最大距离和次大距离,然后更新每个子节点的最大,次大距离,就是把每个子节点当一次根节点,如果它在父节点的最大距离边上,就用次大距离+到父节点的距离与自己的最大次大距离比较,,,,






    #include<stdio.h>
    #include<string.h>
    #define N 10001
    int n,head[N],num,vis[N],dp[N][2];
    struct edge
    {
        int st,ed,next,w;
    }E[N*2];
    void addedge(int x,int y,int w)
    {
        E[num].st=x;
        E[num].ed=y;
        E[num].w=w;
        E[num].next=head[x];
        head[x]=num++;
    }
    int dfs(int u)
    {
        int i,v;
        vis[u]=1;
        for(i=head[u];i!=-1;i=E[i].next)
        {
            v=E[i].ed;
            if(vis[v]==1)continue;
            int temp=dfs(v)+E[i].w;
            if(temp>dp[u][1])//最长边
            {
                dp[u][0]=dp[u][1];
                dp[u][1]=temp;
            }
            else if(temp>dp[u][0])//次长边
            {
                dp[u][0]=temp;
            }
        }
        return dp[u][1]; 
    }
    void dfs1(int u)
    {
      vis[u]=1;
      int i,v,temp;
      for(i=head[u];i!=-1;i=E[i].next)
      {
          v=E[i].ed;
          if(vis[v]==1)continue;
          if(dp[v][1]+E[i].w==dp[u][1])//如果改点在父节点的最长边上,就与次长边比较
            temp=dp[u][0]+E[i].w;
          else temp=dp[u][1]+E[i].w;//不在父节点的最长边上,就与最长边比较
            if(temp>dp[v][1])
            {
                dp[v][0]=dp[v][1];
                dp[v][1]=temp;
            }
            else if(temp>dp[v][0])
            {
                dp[v][0]=temp;
            }
            dfs1(v);
      }
    }
    int main()
    {
        int i,x,y,w;
        while(scanf("%d",&n)!=-1)
        {
            memset(head,-1,sizeof(head));
            num=0;
            for(x=2;x<=n;x++)
            {
                scanf("%d%d",&y,&w);
                addedge(x,y,w);
                addedge(y,x,w);
            }
            memset(dp,0,sizeof(dp));
            memset(vis,0,sizeof(vis));
            dfs(1);
            memset(vis,0,sizeof(vis));
            dfs1(1);
             for(i=1;i<=n;i++)
                 printf("%d
    ",dp[i][0]>dp[i][1]?dp[i][0]:dp[i][1]);
        }
        return 0;
    }



  • 相关阅读:
    实验 4:Open vSwitch 实验——Mininet 中使用 OVS 命令
    实验 3:Mininet 实验——测量路径的损耗率
    福州大学软件工程实践个人编程作业
    实验 2:Mininet 实验——拓扑的命令脚本生成
    实验 1:Mininet 源码安装和可视化拓扑工具
    2020软件工程实践第一次作业
    魏迟燕的自走棋
    中位数之最
    A Simple Math Problem
    数字的升级准则
  • 原文地址:https://www.cnblogs.com/riskyer/p/3228527.html
Copyright © 2011-2022 走看看