题意:

给出一颗有根树,每个顶点上有一个权值\(w_i\)。求前\(k\)大条链的权值之和。

分析:

首先\(dfs\)预处理一遍\(d(u)\),表示从顶点\(u\)往下走能得到的最大权值和,顺便还要记录这是从\(u\)的哪个儿子转移上来的。
最开始我们取树中最大的一条链,取完以后树会分裂成若干不相交的子树。然后再从这些子树中取一条能得到最大权值的链,然后继续分裂。
每次取走一条链后,新出现的子树的根节点是链上所有点的没被取走的儿子节点。
这个过程用一个优先队列维护即可。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <map>
#define MP make_pair
using namespace std;

typedef long long LL;
typedef pair<LL, int> PII;
const int maxn = 100000 + 10;

struct Edge
{
    int v, nxt;
    Edge() {}
    Edge(int v, int nxt):v(v), nxt(nxt) {}
};

int root, ecnt, head[maxn];
Edge edges[maxn];

int n, k;
LL w[maxn];
int indeg[maxn];

void init() {
    ecnt = 0;
    memset(head, 0, sizeof(head));
    memset(indeg, 0, sizeof(indeg));
}

void AddEdge(int u, int v) {
    edges[++ecnt] = Edge(v, head[u]);
    head[u] = ecnt;
}

LL d[maxn];
int Mson[maxn];

void dfs(int u) {
    d[u] = w[u];
    LL Max = -1;
    for(int i = head[u]; i; i = edges[i].nxt) {
        int v = edges[i].v;
        dfs(v);
        if(d[v] > Max) { Max = d[v]; Mson[u] = v; }
    }
    if(Max >= 0) d[u] += Max;
}

priority_queue<PII> Q;

int main()
{
    int T; scanf("%d", &T);
    for(int kase = 1; kase <= T; kase++) {
        init();
        scanf("%d%d", &n, &k);
        for(int i = 1; i <= n; i++) scanf("%lld", w + i);
        for(int i = 1; i < n; i++) {
            int u, v; scanf("%d%d", &u, &v);
            AddEdge(u, v);
            indeg[v]++;
        }

        for(int i = 1; i <= n; i++) if(!indeg[i]) { root = i; break; }

        memset(Mson, 0, sizeof(Mson));
        dfs(root);
        while(!Q.empty()) Q.pop();
        Q.push(MP(d[root], root));
        LL ans = 0;
        for(int i = 0; i < k && !Q.empty(); i++) {
            PII x = Q.top(); Q.pop();
            ans += x.first;
            int u = x.second;
            for(; u; u = Mson[u]) {
                for(int j = head[u]; j; j = edges[j].nxt) {
                    int v = edges[j].v;
                    if(v != Mson[u]) Q.push(MP(d[v], v));
                }
            }
        }

        printf("Case #%d: %lld\n", kase, ans);
    }

    return 0;
}