HDU 2196 Computer 二次扫描与换根DP

题意:给定一棵树,求树上所有点到其最远点的距离。

数据范围: 1 <= N <= 100000

------------------------------------------我是分割线------------------------------------------

题解:对于每个节点u来说,其可能到达的最长距离为max{其子树内的最长距离,其父节点不经过u的子树内的最长距离}。于是,我们便可以在第一遍dfs中预处理节点x到其子树内的最长距离,顺带求一下次长距离,方便转移。

// f[x][0] 表示x到其子树中叶子节点的最长距离 ,f[x][1] 表示x到其子树中叶子节点的次长距离。 
void dfs1(int x, int fa){
    for(int i = head[x]; i; i = e[i].next) {
        int y = e[i].to;
        if(y == fa) continue;
        dfs1(y, x);
        if(f[y][0] + e[i].v > f[x][0]){ //更新最长距离和次长距离。
            f[x][1] = f[x][0]; 
            f[x][0] = f[y][0] + e[i].v;
        }
        else f[x][1] = max(f[x][1], f[y][0] + e[i].v);
    }
}

在第二次扫描中,考虑每个点换根所改变的贡献。

分两种情况:①当前节点v不在父节点的最长路径上.②当前节点v在父节点的最长路径上.

于是状态转移方程显然:① d[y] = max(f[x][0], d[x]) + e[i].v; ② d[y] = max(f[x][1], d[x]) + e[i].v;

总时间复杂度O(N)。

void dfs2(int x, int fa){
    for(int i = head[x]; i; i = e[i].next){
        int y = e[i].to;
        if(y == fa) continue;
        if(f[x][0] == f[y][0] + e[i].v) d[y] = max(f[x][1], d[x]) + e[i].v; // y在x最长距离的路径上。 
        else d[y] = max(f[x][0], d[x]) + e[i].v; // y不在x最长距离的路径上。 
        ans[y] = max(d[y], f[y][0]);    
        dfs2(y, x);
    }
}

于是,这道题就解决了,附上完整代码:

#include<bits/stdc++.h>

#define ll long long
#define mp make_pair
#define rep(i, a, b) for(int i = (a); i <= (b); i++)
#define per(i, a, b) for(int i = (a); i >= (b); i--)

using namespace std;

typedef pair<int, int> pii;
typedef double db;
const int N = 1e6 + 50;
int n, head[N], cnt = 0, ans[N];
int f[N][2], d[N];
struct node{ int to, next, v; } e[N]; 
inline int read(){
    int x = 0, f = 1;
    char ch = getchar();
    while(ch < '0' || ch > '9') { if(ch == '-') f = -1; ch = getchar();}
    while(ch >='0' && ch <='9') { x = (x<<3)+(x<<1)+(ch^48); ch = getchar();}
    return x*f;
}
void add(int x, int y, int z) {
    e[++cnt].to = y; e[cnt].v = z;
    e[cnt].next = head[x]; head[x] = cnt;
}
// f[x][0] 表示x到其子树中叶子节点的最长距离 ,f[x][1] 表示x到其子树中叶子节点的次长距离。 
void dfs1(int x, int fa){
    for(int i = head[x]; i; i = e[i].next) {
        int y = e[i].to;
        if(y == fa) continue;
        dfs1(y, x);
        if(f[y][0] + e[i].v > f[x][0]){ //状态转移 
            f[x][1] = f[x][0]; 
            f[x][0] = f[y][0] + e[i].v;
        }
        else f[x][1] = max(f[x][1], f[y][0] + e[i].v);
    }
}
void dfs2(int x, int fa){
    for(int i = head[x]; i; i = e[i].next){
        int y = e[i].to;
        if(y == fa) continue;
        if(f[x][0] == f[y][0] + e[i].v) d[y] = max(f[x][1], d[x]) + e[i].v; // y在x最长距离的路径上。 
        else d[y] = max(f[x][0], d[x]) + e[i].v; // y不在x最长距离的路径上。 
        ans[y] = max(d[y], f[y][0]);    
        dfs2(y, x);
    }
}
void init(){
    n = read();
    rep(i, 1, n-1) {
        int yy = read(), vv = read();
        add(i+1, yy, vv); add(yy, i+1, vv);
    }
    dfs1(1, 0); ans[1] = f[1][0];
    dfs2(1, 0);
    rep(i, 1, n) printf("%d\n", ans[i]);
}
int main(){
    init();
    return 0;
}
View Code

 

posted @ 2019-10-01 15:48  smilke  阅读(209)  评论(0编辑  收藏  举报