「题解」 P2619 [国家集训队] Tree I

AtomAlpaca

Table of contents

题目

给出一个无向带权连通图,每条边是黑色或白色。求一棵最小权的恰好有 needneed 条白色边的生成树。

link

题解

凸性的证明思路和最小度限制生成树类似。首先对于每个白边 (u,v,w)(u, v, w),考虑新增一条黑边 (u,v,+)(u, v, +\infty),防止白边过少时无法得到生成树。考虑当前有 k1k - 1 条白边的生成树,现在要强行加进去一个白边,设其权值为 w1w_1,这使得原来的生成树上多出来一个环,我们选这个环上所有黑边中权值最大的一个删掉,设其权值为 w2w_2。我们每次要选择加入后构成的环中有黑边,且权值增量 w=w1w2w = w_1 - w_2 最小的一个白边。不难发现这两个限制随着白边的加入都是越来越紧的——第 kk 次未加入生成树的白边第 k+1k + 1 次可能加入了,第 kk 次存在的黑边第 k+1k + 1 次可能不存在。

因此如果第 k+1k + 1 次加入时的增量小于 kk 次,那一定可以交换 k,k+1k, k + 1 次的选边方案使得第 kk 次更优,因此加入白边后的权值增量不增,故该问题有凸性,可以用 wqs 二分解决。

于是二分一个增量,每次将所有白边权值加上这个增量,再和黑边一起跑最小生成树,根据最小生成树中白边的数量调整增量即可。

代码

#include <bits/stdc++.h>

using std::cin;
using std::cout;

const int MAX = 1e5 + 5;

int n, m, k, u, v, w, c, cnt, res, ans;
int f[MAX];
struct E { int u, v, w, c; } e[MAX];

int find(int x) { if (f[x] == x) { return x; } return f[x] = find(f[x]); }
bool cmp(E a, E b) { if (a.w == b.w) { return a.c < b.c; } return a.w < b.w; }

void check(int x)
{
    res = cnt = 0;
    for (int i = 1; i <= m; ++i) { if (e[i].c == 0) { e[i].w += x; } }
    for (int i = 1; i <= n; ++i) { f[i] = i; }
    std::sort(e + 1, e + m + 1, cmp);
    for (int i = 1, tmp = 0; i <= m and tmp != n - 1; ++i)
    {
        int fu = find(e[i].u), fv = find(e[i].v);
        if (fu == fv) { continue; }
        f[fv] = fu; if (e[i].c == 0) { ++cnt; } res += e[i].w; ++tmp;
    }
    for (int i = 1; i <= m; ++i) { if (e[i].c == 0) { e[i].w -= x; } }
}

int main()
{
    cin >> n >> m >> k;
    for (int i = 1; i <= m; ++i)
    {
        cin >> u >> v >> w >> c;
        e[i] = {u + 1, v + 1, w, c};
    }
    int l = -111, r = 111, mid = 0;
    while (l <= r)
    {
        mid = (l + r) >> 1;
        check(mid);
        if (cnt < k) { r = mid - 1; } else { l = mid + 1; ans = res - k * mid; }
    }
    cout << ans;
    return 0;
}