zoukankan      html  css  js  c++  java
  • 【cf1241E】E. Paint the Tree(贪心)

    传送门

    题意:
    给出一颗带权树,对于每个点,至多选择(k)个和其相连的边,询问最后选择边的权值和最大为多少。

    思路:

    • 我们用(f_{i,0})表示不选(i)往父亲这条边,(f_{i,1})表示要选。
    • 显然对于一个点(u)而言,我们要选择若干连向儿子的边,假设对于所有儿子(v),我们采用(f_{v,0})。那么现在要选若干儿子,那么贡献则为(f_{v,1}-f_{v,0}+w(u,v))。我们按照这一贡献贪心来选即可。
    • 注意判断贡献是否大于(0)

    代码如下:

    /*
     * Author:  heyuhhh
     * Created Time:  2020/3/12 11:28:02
     */
    #include <iostream>
    #include <algorithm>
    #include <cstring>
    #include <vector>
    #include <cmath>
    #include <set>
    #include <map>
    #include <queue>
    #include <iomanip>
    #include <assert.h>
    #define MP make_pair
    #define fi first
    #define se second
    #define pb push_back
    #define sz(x) (int)(x).size()
    #define all(x) (x).begin(), (x).end()
    #define INF 0x3f3f3f3f
    #define Local
    #ifdef Local
      #define dbg(args...) do { cout << #args << " -> "; err(args); } while (0)
      void err() { std::cout << '
    '; }
      template<typename T, typename...Args>
      void err(T a, Args...args) { std::cout << a << ' '; err(args...); }
      template <template<typename...> class T, typename t, typename... A> 
      void err(const T <t> &arg, const A&... args) {
      for (auto &v : arg) std::cout << v << ' '; err(args...); }
    #else
      #define dbg(...)
    #endif
    using namespace std;
    typedef long long ll;
    typedef pair<int, int> pii;
    //head
    const int N = 5e5 + 5;
    
    int n, k;
    struct Edge {
        int v, w, next;  
    }e[N << 1];
    int head[N], tot;
    void adde(int u, int v, int w) {
        e[tot].v = v, e[tot].w = w, e[tot].next = head[u], head[u] = tot++;   
    }
    
    ll dp[N][2];
    
    void dfs(int u, int fa) {
        ll cur = 0;
        vector <ll> adds;
        for(int i = head[u]; i != -1; i = e[i].next) {
            int v = e[i].v, w = e[i].w;
            if(v != fa) {
                dfs(v, u);
                cur += dp[v][0];
                adds.push_back(dp[v][1] + w - dp[v][0]);
            }   
        }
        sort(all(adds));
        reverse(all(adds));
        for(int i = 0; i < min(sz(adds), k); i++) if(adds[i] > 0) {
            cur += adds[i];
        }
        dp[u][0] = dp[u][1] = cur;
        if(k - 1 < sz(adds) && adds[k - 1] > 0) {
            dp[u][1] -= adds[k - 1];
        }
    }
    
    void run() {
        cin >> n >> k;
        for(int i = 1; i <= n; i++) head[i] = -1; tot = 0;
        for(int i = 1; i < n; i++) {
            int u, v, w; cin >> u >> v >> w;
            adde(u, v, w), adde(v, u, w);   
        }
        dfs(1, 0);
        ll ans = dp[1][0];
        cout << ans << '
    '; 
    }
    
    int main() {
        ios::sync_with_stdio(false);
        cin.tie(0); cout.tie(0);
        cout << fixed << setprecision(20);
        int T; cin >> T;
        while(T--) run();
        return 0;
    }
    
  • 相关阅读:
    poj 2718 Smallest Difference
    AtCoder Beginner Contest 100 2018/06/16
    aoj 0009 Prime Number
    poj 1930 Dead Fraction
    poj 3669 Meteor Shower
    aoj 0121 Seven Puzzle
    poj 2429 GCD & LCM Inverse
    aoj 0005 GCD and LCM
    aoj 0558 Cheese
    aoj 0033 玉
  • 原文地址:https://www.cnblogs.com/heyuhhh/p/12506964.html
Copyright © 2011-2022 走看看