hdu 2196 computer

题意

给你一棵树,边有权值。

对于每一个点,求其与其距离最远的点的距离。

分析

思路1:树的直径

利用直径的性质进行求解,网上资料很多,这里不赘述。

#include <cstdio>
#include <cstring>
#include <cctype>
#include <algorithm>
using namespace std;

const int N=16384;

int n;
struct G
{
    int v,d,nxt;
}mp[N<<1];
int tt,hd[N];

int dis[N];
int vis[N];
int mx[N];

inline int read(void)
{
    int x=0; char c=getchar();
    for (;!isdigit(c);c=getchar());
    for (;isdigit(c);c=getchar()) x=x*10+c-'0';
    return x;
}

inline void ins(int u,int v,int d)
{
    mp[++tt].v=v;
    mp[tt].d=d;
    mp[tt].nxt=hd[u];
    hd[u]=tt;
}

void dfs(int now)
{
    vis[now]=1;
    for (int k=hd[now];k;k=mp[k].nxt)
        if (!vis[mp[k].v])
        {
            dis[mp[k].v]=dis[now]+mp[k].d;
            dfs(mp[k].v);
        }
}

int main(void)
{
//  freopen("a.in","r",stdin);
//  freopen("a.out","w",stdout);

    while (scanf("%d",&n)!=EOF)
    {
        int x,y;
        memset(hd,0,sizeof hd); tt=0;
        for (int i=2;i<=n;i++)
        {
            x=read(),y=read();
            ins(i,x,y),ins(x,i,y);
        }

        int tmp,tvt;
        memset(mx,0,sizeof mx);
        memset(dis,0,sizeof dis);
        memset(vis,0,sizeof vis);
        dfs(1);
        for (int i=1;i<=n;i++)
            mx[i]=max(mx[i],dis[i]);

        tvt=1,tmp=mx[1];
        for (int i=2;i<=n;i++)
            if (tmp<mx[i]) tvt=i,tmp=mx[i];
        memset(dis,0,sizeof dis);
        memset(vis,0,sizeof vis);
        dfs(tvt);
        for (int i=1;i<=n;i++)
            mx[i]=max(mx[i],dis[i]);

        tvt=1,tmp=mx[1];
        for (int i=2;i<=n;i++)
            if (tmp<mx[i]) tvt=i,tmp=mx[i];
        memset(dis,0,sizeof dis);
        memset(vis,0,sizeof vis);
        dfs(tvt);
        for (int i=1;i<=n;i++)
            mx[i]=max(mx[i],dis[i]);

        for (int i=1;i<=n;i++)
            printf("%d\n",mx[i]);
    }

    return 0;
}

思路2:二次树形dp

我们要求每个点与其最远点的距离。

我们随便选择一个点构造一棵有根树,那么对于任意一个点,它的直径无非两种情况:

①点向的子树内延伸最长的长度

②点向的子树外延伸最长的长度

对于第一种情况,我们需要记录:从点i往下延伸的最长长度

对于第二种情况,我们需要记录:从点i往外延伸的最长长度

设的父亲为,一种情况是在点时弯了,另一种情况是点往外延伸。

所以

然后需要优化处理。

方法1:从前往后扫一遍,从后往前扫一遍。

方法2:记录最大值和次大值,若i是pre的最大值,那么就取次大,否则取最大。

#include <cstdio>
#include <cstring>
#include <cctype>
#include <algorithm>
using namespace std;

#define rep(i,a,b) for (int i=(a);i<=(b);i++)
#define per(i,a,b) for (int i=(a);i>=(b);i--)
#define fore(k,u) for (int k=hd[u];k>0;k=mp[k].nx)

#define ed lis[i]

const int N=16384;
const int E=32768;

int n;
struct G
{
    int v,d;
    int nx;

    inline G(int _v=0,int _d=0,int _nx=0)
    {
        v=_v,d=_d;
        nx=_nx;
    }
}mp[E];
int tot,hd[N];

int fx[N];
int ux[N]; int lis[E],len;

int ans[N];

inline int rd(void)
{
    int x=0,f=1; char c=getchar();
    for (;!isdigit(c);c=getchar()) if (c=='-') f=-1;
    for (;isdigit(c);c=getchar()) x=x*10+c-'0';
    return x*f;
}

inline void Ins(int u,int v,int d)
{
    mp[++tot]=G(v,d,hd[u]); hd[u]=tot;
    mp[++tot]=G(u,d,hd[v]); hd[v]=tot;
}

void FirDFS(int now,int pre)
{
    fore(k,now) if (mp[k].v!=pre)
        FirDFS(mp[k].v,now);
    fore(k,now) if (mp[k].v!=pre)
        fx[now]=max(fx[now],fx[mp[k].v]+mp[k].d);
}

void SecDFS(int now,int pre)
{
    len=0;
    fore(k,now) if (mp[k].v!=pre)
        lis[++len]=k;

    if (~pre)
    {
        rep(i,1,len)
            ux[mp[ed].v]=max(ux[mp[ed].v],ux[now]);
    }

    int mx=0;
    rep(i,1,len)
    {
        ux[mp[ed].v]=max(ux[mp[ed].v],mx);
        mx=max(mx,fx[mp[ed].v]+mp[ed].d);
    }

    mx=0;
    per(i,len,1)
    {
        ux[mp[ed].v]=max(ux[mp[ed].v],mx);
        mx=max(mx,fx[mp[ed].v]+mp[ed].d);
    }

    rep(i,1,len)
        ux[mp[ed].v]+=mp[ed].d;

    fore(k,now) if (mp[k].v!=pre)
        SecDFS(mp[k].v,now);
}

int main(void)
{
//  freopen("hdu2196.in","r",stdin);
//  freopen("hdu2196.out","w",stdout);

    while (~scanf("%d",&n))
    {
        tot=0; memset(hd,0,sizeof hd);
        rep(i,2,n)
        {
            int v=rd(),d=rd();
            Ins(i,v,d);
        }

        memset(fx,0,sizeof fx);
        memset(ux,0,sizeof ux);
        FirDFS(1,-1);
        SecDFS(1,-1);

        memset(ans,0,sizeof ans);
        rep(i,1,n) ans[i]=max(fx[i],ux[i]);
        rep(i,1,n) printf("%d\n",ans[i]);
    }

    return 0;
}

小结

(1)树的常见方法

①树形dp

②点分治

③树链剖分,dfs序剖分:树–>链

④树上倍增

⑤树上莫队

⑥LCT

当然还要用一些什么树的直径之类的性质。

【hdu2196】Computer的更多相关文章

  1. 【题解】Computer Network

    Description 给你一棵N(N<=10000)个节点的树,求每个点到其他点的最大距离. Input 第一行一个数N.接下来若干行每行两个数k,t描述一条点k到点t的边(输入数据保证无重复 ...

  2. 【HDOJ2196】Computer(树的直径,树形DP)

    题意:给定一棵N个点树,询问这个树里面每个点到树上其他点的最大距离. n<=10000 思路:设f[u,1],f[u,2]为以U为根向下的最长与次长,g[u,1],g[u,2]为从哪个儿子转移来 ...

  3. Codeforces 716A Crazy Computer 【模拟】 (Codeforces Round #372 (Div. 2))

    A. Crazy Computer time limit per test 2 seconds memory limit per test 256 megabytes input standard i ...

  4. 【HDU 2196】 Computer(树的直径)

    [HDU 2196] Computer(树的直径) 题链http://acm.hdu.edu.cn/showproblem.php?pid=2196 这题可以用树形DP解决,自然也可以用最直观的方法解 ...

  5. 【HDU 2196】 Computer (树形DP)

    [HDU 2196] Computer 题链http://acm.hdu.edu.cn/showproblem.php?pid=2196 刘汝佳<算法竞赛入门经典>P282页留下了这个问题 ...

  6. 【HDU 2196】 Computer

    [题目链接] 点击打开链接 [算法] 我们知道,一棵树上离某个节点最远的节点,可能是经过它的祖先,再到那个祖先的某个孩子,或者,是它的那颗子树中,离它最远的一个节点,就不难想到以下算法 : 第一遍DF ...

  7. 企业IT管理员IE11升级指南【3】—— IE11 新的GPO设置

    企业IT管理员IE11升级指南 系列: [1]—— Internet Explorer 11增强保护模式 (EPM) 介绍 [2]—— Internet Explorer 11 对Adobe Flas ...

  8. 【转】linux和windows下安装python集成开发环境及其python包

    本系列分为两篇: 1.[转]windows和linux中搭建python集成开发环境IDE 2.[转]linux和windows下安装python集成开发环境及其python包 3.windows和l ...

  9. 【转载】如何破解受保护的excel密码

    [工具] 1.电脑一台(安装有Microsoft Excel) 2.受保护excel一个 [步骤] 1.首先,打开受保护的Excel表格,按"ALT"+"F11" ...

随机推荐

  1. for循环嵌套的穷举,迭代,以及while的使用和for的转换

    for循环的穷举.解决多种方法做种组合问题,代替人脑的大量计算 穷举例子 迭代,寻找一定的规律.然后利用循环找出结果 迭代举例 以五个小朋友3岁,之前的每一个小朋友都比序号在后的小朋友大两岁.温,第一 ...

  2. dubbo源码之三——dubbo重构

    dubbo版本:2.5.4 转自:http://javatar.iteye.com/blog/1041832

  3. HDU4549 M斐波那契数列 矩阵快速幂+欧拉函数+欧拉定理

    M斐波那契数列 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Total Sub ...

  4. .Net鼠标随动窗口

    就像QQ宠物或者迅雷悬浮窗口一样,鼠标点下去窗体跟着鼠标动 主要是两个时间的加载 MouseDown和MouseMove事件 MouseDown事件: private int _StartX ;//鼠 ...

  5. Q查询

    一.Complex lookups with Q objects(Q对象的复杂查询) 仅仅靠单一的关键字参数查询已经很难满足查询要求.此时Django为我们提供了Q查询: class Q 1.Q对象( ...

  6. C#窗体->>随机四则运算

    用户需求: 程序能接收用户输入的整数答案,并判断对错程序结束时,统计出答对.答错的题目数量.补充说明:0——10的整数是随机生成的用户可以选择四则运算中的一种用户可以结束程序的运行,并显示统计结果.在 ...

  7. 5.mybatis一对一表关联查询

    方式一:嵌套结果:使用嵌套结果映射来处理重复的联合结果的子集,封装联表查询的数据(去除重复的数据)  SELECT * FROM class c,teacher t WHERE c.tid = t.t ...

  8. Linux命令工具基础02 文件及目录管理

    文件及目录管理 文件管理不外乎文件或目录的创建.删除.查询.移动,有mkdir/rm/mv 文件查询是重点,用find来进行查询:find的参数丰富,也非常强大: 查看文件内容是个大的话题,文本的处理 ...

  9. shiro连接数据库

    建一个jdbcRealm.ini [main] jdbcRealm=org.apache.shiro.realm.jdbc.JdbcRealm dataSource=com.mchange.v2.c3 ...

  10. Nexus4_文件名乱码

    1. 官方的出厂映像 for Android4.4:occam-krt16s-factory-2006f418.tgz 2. 自己编译的 Android-4.4_r1 (AOSP on Mako) 映 ...