树网的核

md, 链式前向星, 因为cost是ll, 就写下面了, 都开 [N << 1]了, 就co[N], 硬是自己瞎想,找了2h bug, 🤮

题面

设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边带有正整数的权,我们称T为树网(treenetwork),其中V, E分别表示结点与边的集合,W表示各边长度的集合,并设T有n个结点。

路径:树网中任何两结点a,b都存在唯一的一条简单路径,用d(a,b)表示以a,b为端点的路径的长度,它是该路径上各边长度之和。

我们称d(a,b)为a,b两结点间的距离。

一点v到一条路径P的距离为该点与P上的最近的结点的距离:

d(v,P)=min{d(v,u),u为路径P上的结点}。

树网的直径:树网中最长的路径称为树网的直径。

对于给定的树网T,直径不一定是唯一的,但可以证明:各直径的中点(不一定恰好是某个结点,可能在某条边的内部)是唯一的,我们称该点为树网的中心。

偏心距ECC(F):树网T中距路径F最远的结点到路径F的距离,即:

ECC(F)=max{d(v,F),v∈V}

任务:对于给定的树网T=(V, E,W)和非负整数s,求一个路径F,它是某直径上的一段路径(该路径两端均为树网中的结点),其长度不超过s(可以等于s),使偏心距ECC(F)最小。

我们称这个路径为树网T=(V,E,W)的核(Core)。

必要时,F可以退化为某个结点。

一般来说,在上述定义下,核不一定只有一个,但最小偏心距是唯一的。

输入格式

包含n行: 第1行,两个正整数n和s,中间用一个空格隔开,其中n为树网结点的个数,s为树网的核的长度的上界,设结点编号依次为1, 2, …, n。

从第2行到第n行,每行给出3个用空格隔开的正整数,依次表示每一条边的两个端点编号和长度。

例如,“2 4 7”表示连接结点2与4的边的长度为7。

所给的数据都是正确的,不必检验。

输出格式

只有一个非负整数,为指定意义下的最小偏心距。

数据范围

n≤500000,s<231

输入样例:

5 2
1 2 5
2 3 2
2 4 4
2 5 3

输出样例:

5

题解

求直径, 回溯之类的就不说了

证明部分

在任意一条直径上求出的最小偏心距都相等

树的直径中点成活, 相当于 任意一条直径 把其他直径切成了 两半

要明白对答案有贡献的时 路径距直径两端的距离 和 路径上分支的最大长度(当直径长度小于等于s时才贡献)

当直径不止一条时, 这偏心距必为直径的一半, 且路径包含直径的中点 (自己画画, 本人画丑就不放图了)

解法一, 枚举 O(\(n^3\))

枚举就不用说了吧, 两层遍历一层搜索

这个只用于 n <= 300 数据范围的题, 5e5 就别考虑了

解法二,枚举+贪心 O(\(n^2\))

在直径上尺取法, 枚举少了一层

解法三, 二分 O(nlogSUM)

答案单调, 自然可二分

解法四和五, 线性 O(n)

解法四就是和五差不多, 一个是分类讨论, 一个是单调队列, 线性思路是一样的

上面证明说了 路径上分支的最大长度对答案贡献(当直径长度小于等于s时才贡献)

方法四就是分 直径大于s和 小于等于 s

方法五就时一股脑全考虑, 由单调队列决定

这里直接上法4, 5代码

#include <bits/stdc++.h>
#define all(n) (n).begin(), (n).end()
#define se second
#define fi first
#define pb push_back
#define mp make_pair
#define sqr(n) (n)*(n)
#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 long long ll;
typedef pair<int, int> PII;
typedef pair<ll, ll> PLL;
typedef vector<int> VI;
typedef double db;

const int N = 5e5 + 5;

int n, m, _, k;
int h[N], ne[N << 1], to[N << 1], tot, f[N];
ll co[N << 1], d[N], b[N], ans = 2e18;
bool v[N];

void add(int u, int v, ll c) {
    ne[++tot] = h[u]; h[u] = tot; to[tot] = v; co[tot] = c;
}

void dfs(int u) {
    v[u] = 1;

    for (int i = h[u]; i; i = ne[i]) {
        int y = to[i];
        if (v[y]) continue;
        d[y] = d[u] + co[i];
        if (d[y] > d[0]) d[0] = d[y], f[0] = y;
        f[y] = u;
        dfs(y);
    }
}

void work(int& p, int& q) {
    memset(v, 0, sizeof v); d[0] = -N;
    d[1] = 0; dfs(1); p = f[0]; f[p] = 0;
    memset(v, 0, sizeof v); d[0] = -N;
    d[p] = 0; dfs(p); q = f[0]; 
}

void dfss(int u, int fa) {
    v[u] = 1;

    for (int i = h[u]; i; i = ne[i]) {
        int y = to[i];
        if (v[y]) continue;

        if (u == fa) d[y] = co[i];
        else d[y] = d[u] + co[i];

        b[fa] = max(d[y], b[fa]);
        dfss(y, fa);
    }
}

int main() {
    ios::sync_with_stdio(0); cin.tie(0);
    cin >> n >> m;
    rep(i, 2, n) {
        int u, v; ll c; cin >> u >> v >> c;
        add(u, v, c); add(v, u, c);
    }

    int p, q; work(p, q); 

    if (d[q] > m) {
        for (int i = q, j = q; i; i = f[i]) {
            while (f[j] && d[i] - d[f[j]] <= m) j = f[j];
            ans = min(ans, max(d[j], d[q] - d[i]));
        }
    }
    else {
        ans = 0;
        memset(v, 0, sizeof v);
        for (int i = q; i; i = f[i]) v[f[i]] = 1, dfss(i, i), ans = max(ans, b[i]);
    }

    cout << ans;
    return 0;
}
#include <bits/stdc++.h>
#define all(n) (n).begin(), (n).end()
#define se second
#define fi first
#define pb push_back
#define mp make_pair
#define sqr(n) (n)*(n)
#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 long long ll;
typedef pair<int, int> PII;
typedef pair<ll, ll> PLL;
typedef vector<int> VI;
typedef double db;

const int N = 5e5 + 5;

int n, m, _, k;
int h[N], ne[N << 1], to[N << 1], tot, f[N], qu[N];
ll co[N << 1], d[N], b[N], ans = 2e18;
bool v[N];

void add(int u, int v, ll c) {
    ne[++tot] = h[u]; h[u] = tot; to[tot] = v; co[tot] = c;
}

void dfs(int u) {
    v[u] = 1;

    for (int i = h[u]; i; i = ne[i]) {
        int y = to[i];
        if (v[y]) continue;
        d[y] = d[u] + co[i];
        if (d[y] > d[0]) d[0] = d[y], f[0] = y;
        f[y] = u;
        dfs(y);
    }
}

void work(int& p, int& q) {
    memset(v, 0, sizeof v); d[0] = -N;
    d[1] = 0; dfs(1); p = f[0];
    memset(v, 0, sizeof v); d[0] = -N;
    d[p] = 0; dfs(p); q = f[0];
}

void dfss(int u, int fa) {
    v[u] = 1;

    for (int i = h[u]; i; i = ne[i]) {
        int y = to[i];
        if (v[y]) continue;

        if (u == fa) d[y] = co[i];
        else d[y] = d[u] + co[i];

        b[fa] = max(d[y], b[fa]);
        dfss(y, fa);
    }
}

int main()
{
    ios::sync_with_stdio(0); cin.tie(0);
    cin >> n >> m;
    rep(i, 2, n) {
        int u, v; ll c; cin >> u >> v >> c;
        add(u, v, c); add(v, u, c);
    }

    int p, q; work(p, q);

    memset(v, 0, sizeof v); f[p] = 0;

    for (int i = q; i; i = f[i]) v[f[i]] = 1, dfss(i, i);

    qu[0] = b[q]; 
    for (int i = q, j = q, h = 0, t = 0; i; i = f[i]) {
        while (f[j] && d[i] - d[f[j]] <= m) {
            j = f[j];
            while (t >= h && b[j] >= b[qu[t]]) --t;
            qu[++t] = j;
        }
        
        if (t >= h) ans = min(ans, max(max(d[q] - d[i], d[j]), b[qu[h]]));
        else ans = min(ans, max(d[q] - d[i], d[j]));
        
        if (qu[h] == i) ++h;
    }

    cout << ans;
    return 0;
}
posted @ 2020-06-27 21:46  洛绫璃  阅读(180)  评论(0编辑  收藏  举报