HDU-4123 Bob’s Race(树型dp+RMQ)

传送门:HDU-4123

题意:有n个点组成一棵树编号为1~n,定义一个点的最远距离为从该点出发的最长简单路径,有m次询问:求最大区间长度,使得区间内所有点最远距离的最大值-最远距离的最小值<=k

题解:先2次dfs求出每个点的最远距离,再用尺取法+RMQ找出最长区间

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int MX  = 5e4 + 5;
const LL mod = 998244353;
struct Edge {
    int v, w, nxt;
} E[MX * 2];
int head[MX], tot, n, m;
void add(int u, int v, int w) {
    E[tot].v = v;
    E[tot].w = w;
    E[tot].nxt = head[u];
    head[u] = tot++;
}
void init() {
    for (int i = 1; i <= n; i++) head[i] = -1;
    tot = 0;
}
int ma[MX], se[MX], id1[MX], id2[MX];
void dfs(int u, int fa) {
    ma[u] = se[u] = id1[u] = id2[u] = 0;
    for (int i = head[u]; ~i; i = E[i].nxt) {
        int v = E[i].v, w = E[i].w;
        if (v == fa) continue;
        dfs(v, u);
        if (ma[v] + w > ma[u]) {
            se[u] = ma[u];
            id2[u] = id1[u];
            ma[u] = ma[v] + w;
            id1[u] = v;
        } else if (ma[v] + w > se[u]) {
            se[u] = ma[v] + w;
            id2[u] = v;
        }
    }
}
void DFS(int u, int fa) {
    for (int i = head[u]; ~i; i = E[i].nxt) {
        int v = E[i].v, w = E[i].w;
        if (v == fa) continue;
        if (id1[u] != v) {
            if (ma[u] + w > ma[v]) {
                se[v] = ma[v];
                id2[v] = id1[v];
                ma[v] = ma[u] + w;
                id1[v] = u;
            } else if (ma[u] + w > se[v]) {
                se[v] = ma[u] + w;
                id2[v] = u;
            }
        } else if (id2[u] != v) {
            if (se[u] + w > ma[v]) {
                se[v] = ma[v];
                id2[v] = id1[v];
                ma[v] = se[u] + w;
                id1[v] = u;
            } else if (se[u] + w > se[v]) {
                se[v] = se[u] + w;
                id2[v] = u;
            }
        }
        DFS(v, u);
    }
}
int f[MX][30][2];
void ST() {
    for (int i = 1; i <= n; i++) f[i][0][0] = f[i][0][1] = ma[i];
    for (int j = 1; (1 << j) <= n; j++) {
        for (int i = 1; i + (1 << j) - 1 <= n; i++) {
            f[i][j][0] = max(f[i][j - 1][0], f[i + (1 << (j - 1))][j - 1][0]);
            f[i][j][1] = min(f[i][j - 1][1], f[i + (1 << (j - 1))][j - 1][1]);
        }
    }
}
int RMQ(int l, int r) {
    int j = 0;
    while ((1 << (j + 1)) <= (r - l + 1)) j++;
    int ans1 = max(f[l][j][0], f[r - (1 << j) + 1][j][0]);
    int ans2 = min(f[l][j][1], f[r - (1 << j) + 1][j][1]);
    return ans1 - ans2;
}
int main() {
    int T;
    //freopen("in.txt", "r", stdin);
    while (scanf("%d%d", &n, &m), n || m) {
        init();
        for (int i = 1, u, v, w; i < n; i++) {
            scanf("%d%d%d", &u, &v, &w);
            add(u, v, w); add(v, u, w);
        }
        dfs(1, -1);
        DFS(1, -1);
        ST();
        while (m--) {
            int k, ans = 0;
            scanf("%d", &k);
            for (int s = 1, t = 1; t <= n; t++) {
                while (s < t && RMQ(s, t) > k) s++;
                ans = max(ans, t - s + 1);
            }
            printf("%d\n", ans);
        }
    }
    return 0;
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值