/ SeriousOJ /

Record Detail

Accepted


  
# Status Time Cost Memory Cost
#1 Accepted 1ms 556.0 KiB
#2 Accepted 1ms 540.0 KiB
#3 Accepted 1ms 584.0 KiB
#4 Accepted 1ms 684.0 KiB
#5 Accepted 1ms 540.0 KiB
#6 Accepted 1ms 560.0 KiB
#7 Accepted 1ms 772.0 KiB
#8 Accepted 1ms 540.0 KiB
#9 Accepted 1ms 540.0 KiB
#10 Accepted 1ms 584.0 KiB
#11 Accepted 1ms 540.0 KiB
#12 Accepted 82ms 6.742 MiB
#13 Accepted 92ms 6.734 MiB
#14 Accepted 82ms 6.77 MiB
#15 Accepted 82ms 6.984 MiB
#16 Accepted 69ms 6.77 MiB
#17 Accepted 78ms 6.711 MiB
#18 Accepted 77ms 6.566 MiB
#19 Accepted 93ms 6.77 MiB
#20 Accepted 77ms 6.566 MiB
#21 Accepted 92ms 6.77 MiB
#22 Accepted 94ms 6.676 MiB
#23 Accepted 88ms 6.574 MiB
#24 Accepted 79ms 6.734 MiB
#25 Accepted 74ms 6.562 MiB
#26 Accepted 72ms 6.77 MiB
#27 Accepted 73ms 6.574 MiB
#28 Accepted 68ms 6.609 MiB
#29 Accepted 95ms 6.582 MiB
#30 Accepted 81ms 6.773 MiB
#31 Accepted 88ms 6.605 MiB
#32 Accepted 93ms 6.734 MiB
#33 Accepted 92ms 6.785 MiB
#34 Accepted 69ms 6.566 MiB
#35 Accepted 76ms 6.57 MiB
#36 Accepted 79ms 6.781 MiB
#37 Accepted 91ms 6.574 MiB
#38 Accepted 72ms 6.77 MiB
#39 Accepted 85ms 6.566 MiB
#40 Accepted 74ms 6.57 MiB
#41 Accepted 91ms 6.566 MiB
#42 Accepted 49ms 6.602 MiB
#43 Accepted 51ms 6.566 MiB
#44 Accepted 43ms 7.195 MiB
#45 Accepted 42ms 7.074 MiB
#46 Accepted 41ms 6.941 MiB
#47 Accepted 146ms 104.508 MiB
#48 Accepted 123ms 55.734 MiB
#49 Accepted 109ms 55.52 MiB
#50 Accepted 60ms 21.02 MiB

Code

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
 
const ll MINF = -1e18;
 
int N, K;
vector<ll> val;
vector<vector<int>> adj;
 
// For each node u, dp[u][d] = maximum sum of a downward path starting at u with exactly d nodes.
 
// The DFS returns a vector dp (size K+1, where index 1..K are used).
// 'ans' is updated globally with any valid path of exactly K nodes.
vector<ll> dfs(int u, int parent, ll &ans) {
    vector<ll> dp(K+1, MINF);
    dp[1] = val[u];  // The trivial path that is just the node u.
    
    // Process every child v of u.
    for (int v : adj[u]) {
        if(v == parent) continue;
        vector<ll> childDP = dfs(v, u, ans);
        
        // --- Combination step ---
        // Before merging the new child's information into dp[u], try to combine
        // a branch already in dp[u] (coming from a different child or u itself)
        // with a new branch from v.
        // A branch from child v gives (when attached to u):
        //   new_branch_length = L+1, with sum = val[u] + childDP[L]
        // Now, if dp[u] already has a branch of length i (which already includes u),
        // combining the two branches (and subtracting one copy of u) gives:
        //   total length = i + (L+1) - 1 = i + L.
        // To have exactly K nodes we require: i + L = K.
        for (int i = 1; i < K; i++) {
            int L = K - i; // L must be at least 1 and at most K-1.
            if(L >= 1 && L <= K-1) {
                if(dp[i] != MINF && childDP[L] != MINF) {
                    ans = max(ans, dp[i] + childDP[L]);
                }
            }
        }
        
        // --- Merging step ---
        // Now update dp[u] with the new downward branches coming from v.
        // For every possible branch from v of length L (1 <= L <= K-1),
        // we can form a branch from u of length L+1 with sum = val[u] + childDP[L].
        vector<ll> newDP = dp;
        for (int L = 1; L <= K-1; L++) {
            if(childDP[L] != MINF) {
                newDP[L+1] = max(newDP[L+1], val[u] + childDP[L]);
            }
        }
        dp = newDP;
    }
    // Also update the answer with any downward branch from u that has exactly K nodes.
    ans = max(ans, dp[K]);
    return dp;
}
 
int main(){
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
 
    cin >> N >> K;
    val.resize(N+1);
    for (int i = 1; i <= N; i++){
        cin >> val[i];
    }
    adj.resize(N+1);
    for (int i = 1; i <= N-1; i++){
        int u, v;
        cin >> u >> v;
        adj[u].push_back(v);
        adj[v].push_back(u);
    }
    
    ll ans = MINF;
    dfs(1, -1, ans);
    cout << (ans == MINF ? 0 : ans) << "\n";
    return 0;
}

Information

Submit By
Type
Submission
Problem
P1161 Max path sum (Hard Version)
Contest
Brain Booster #8
Language
C++17 (G++ 13.2.0)
Submit At
2025-02-17 16:58:56
Judged At
2025-02-17 16:58:56
Judged By
Score
100
Total Time
146ms
Peak Memory
104.508 MiB