Chgtaxihe's blog Chgtaxihe's blog
首页
练习
算法学习
读书笔记
小玩意

Chgtaxihe

首页
练习
算法学习
读书笔记
小玩意
  • 训练补题
  • Codeforces 刷题计划
  • 概率dp练习
  • dp专题练习
  • 数论练习
  • 搜索专题练习
  • 图论练习
  • 基础字符串练习
  • AtCoder练习
  • 题解们
  • solutions

  • 训练补题

    • 训练补题-个人1
    • 训练补题-个人10
    • 训练补题-个人2
    • 训练补题-个人5
    • 训练补题-个人6
    • 训练补题-个人8
    • 训练补题-个人9
    • 训练补题-牛客训练赛1
    • 训练补题-牛客训练赛10
    • 训练补题-牛客训练赛2
    • 训练补题-牛客训练赛3
    • 训练补题-牛客训练赛6
    • 训练补题-牛客训练赛6
    • 训练补题-牛客训练赛7
    • 训练补题-牛客训练赛8
    • 训练补题-牛客训练赛9
      • J: The Escape Plan of Groundhog
      • B: Groundhog and Apple Tree
    • 训练补题-杭电多校10
    • 训练补题-杭电多校2
    • 训练补题-杭电多校3
    • 训练补题-杭电多校6
    • 训练补题-杭电多校9
    • NWERC 2018-训练补题11
    • SpbKOSHP 19-训练补题12
    • Gym102576-训练补题13
    • GCPC2018-训练补题14
    • Gym102202-训练补题15

训练补题-牛客训练赛9

# 牛客训练赛9补题记录

# J: The Escape Plan of Groundhog

做法:枚举上下两条边界,接着从左往右做前缀和!

理论AC

# B: Groundhog and Apple Tree

显然影响答案的只有遍历子节点的顺序。

对于子节点x,yx,yx,y,

  • 如果访问后血量都增加(delta[x]>0,delta[y]>0delta[x]>0,delta[y]>0delta[x]>0,delta[y]>0),则先访问需求(need[i]need[i]need[i])较小的
  • 如果访问后血量都减少,则需要考虑不同顺序时需要的血量
    • 先访问xxx,则所需血量为need[x]+min⁡(0,need[y]−(need[x]+delta[x]))need[x]+\min(0,need[y]-(need[x]+delta[x]))need[x]+min(0,need[y]−(need[x]+delta[x]))
    • 先访问yyy,则所需血量为need[y]+min⁡(0,need[x]−(need[y]+delta[y]))need[y]+\min(0,need[x]-(need[y]+delta[y]))need[y]+min(0,need[x]−(need[y]+delta[y]))
  • 否则,先访问加血的节点,再访问扣血的节点
#include <bits/stdc++.h>
#define ll long long
#define ull unsigned long long
#define int128 __int128_t
#define Android ios::sync_with_stdio(false), cin.tie(NULL)
#define redirect_input freopen("./input.txt", "r", stdin);
#define redirect_output freopen("./output.txt", "w", stdout);
#define debug(s, r) std::cerr << #s << ": " << (s) << (r==0?' ':'\n')
#define pii pair<ll, ll>
#define sqr(x) ((x)*(x))
using namespace std;

const ll inf = 0x3f3f3f3f;
const ll mod = 1e9 + 7;
const int maxn = 1e5 + 100;

ll apple[maxn];
ll n;
vector<pii> G[maxn];
ll need[maxn], delta[maxn];

void dfs(int u, int fa){
    need[u] = 0;
    for(pii & e:G[u]){
        if(e.first == fa) continue;
        dfs(e.first, u);
        delta[e.first] -= 2 * e.second;
        need[e.first] += e.second;
        if(need[e.first] < -delta[e.first]){
            need[e.first] = -delta[e.first];
        }
    }

    sort(G[u].begin(), G[u].end(), [&](pii a, pii b){
        int x = a.first, y = b.first;
        if(delta[x] >= 0 && delta[y] >= 0) return need[x] < need[y];
        if(delta[x] < 0 && delta[y] < 0){
            return need[x] + max(0ll, need[y] - (need[x] + delta[x])) <
                   need[y] + max(0ll, need[x] - (need[y] + delta[y]));
        }
        return delta[x] > delta[y];
    });

    ll cur = delta[u] = apple[u];
    for(pii e:G[u]){
        int v = e.first;
        if(v == fa) continue;
        if(cur < need[v]) {
            need[u] += need[v] - cur;
            cur = need[v];
        }
        cur += delta[v];
        delta[u] += delta[v];
        if(cur < 0) {
            need[u] += -cur;
            cur = 0;
        }
    }
    need[u] = max(need[u], -delta[u]);
}

void solve(){
    cin >> n;
    for(int i=0; i<=n; i++) G[i].clear();
    for(int i=1; i<=n; i++) cin >> apple[i];
    for(int i=1, u, v, w; i<n; i++){
        cin >> u >> v >> w;
        G[u].push_back({v, w});
        G[v].push_back({u, w});
    }
    dfs(1, 0);
    cout << need[1] << '\n';
}

signed main(){
    Android;
    int t; cin >> t;
    while(t--)
    solve();
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
上次更新: 2021/02/24, 03:37:30
训练补题-牛客训练赛8
训练补题-杭电多校10

← 训练补题-牛客训练赛8 训练补题-杭电多校10→

最近更新
01
深入理解Java虚拟机
03-04
02
DNS工作原理
02-27
03
改用VuePress啦
02-23
更多文章>
Theme by Vdoing | Copyright © 2019-2021
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式