This submission is migrated from previous version of oj.uz, which used different machine for grading. This submission may have different result if resubmitted.
#include <bits/stdc++.h>
#include <ext/pb_ds/assoc_container.hpp>
//#pragma GCC optimize("Ofast")
//#pragma GCC optimize("avx2")
using namespace std;
using namespace __gnu_pbds;
#define fi first
#define se second
#define all(m) (m).begin(), (m).end()
#define rall(m) (m).rbegin(), (m).rend()
#define vec vector
#define sz(a) (int) (a).size()
#define mpp make_pair
#define mtt make_tuple
typedef long long ll;
typedef long double ld;
typedef unsigned long long ull;
typedef pair <ll, ll> pll;
typedef pair <int, int> pii;
typedef tuple <int, int, int> tui;
template <typename T>
using prq = priority_queue <T>;
template <typename T>
using pgq = priority_queue <T, vec <T>, greater <T>>;
template <typename T> bool umin(T &a, T b) { return a > b ? a = b, 1 : 0; }
template <typename T> bool umax(T &a, T b) { return a < b ? a = b, 1 : 0; }
struct segt{
int n;
vec <ll> t, p;
segt() {}
segt(int n): n(n), t(4 * n), p(4 * n) {}
void build(int tl, int tr, int v, vec <ll> &d){
// cout << "BUILD" << tl << " " << tr << "\n";
assert(v < sz(t));
if (tl + 1 == tr) return void(t[v] = d[tl]);
int tm = tl + tr >> 1;
build(tl, tm, v * 2 + 1, d);
build(tm, tr, v * 2 + 2, d);
t[v] = max(t[v * 2 + 1], t[v * 2 + 2]);
}
void build(vec <ll> &d){
assert(n == sz(d));
if (!d.empty()){
build(0, n, 0, d);
}
}
void apply(int v, ll x){
t[v] += x, p[v] += x;
}
void push(int v){
if (p[v]){
for (int i: {v * 2 + 1, v * 2 + 2}){
apply(i, p[v]);
}
p[v] = 0;
}
}
void upd(int l, int r, ll x, int tl, int tr, int v){
// cout << "UPD " << tl << " " << tr << "\n";
assert(v < sz(t));
if (l >= r) return;
if (tl == l && tr == r) return void(apply(v, x));
push(v);
int tm = tl + tr >> 1;
upd(l, min(tm, r), x, tl, tm, v * 2 + 1);
upd(max(l, tm), r, x, tm, tr, v * 2 + 2);
t[v] = max(t[v * 2 + 1], t[v * 2 + 2]);
}
void upd(int l, int r, ll x){
if (n) upd(l, r, x, 0, n, 0);
}
ll get(int l, int r, int tl, int tr, int v){
// cout << "GET " << tl << " " << tr << "\n";
assert(v < sz(t));
if (l >= r) return 0LL;
if (tl == l && tr == r) return t[v];
push(v);
int tm = tl + tr >> 1;
return max(get(l, min(tm, r), tl, tm, v * 2 + 1),
get(max(l, tm), r, tm, tr, v * 2 + 2));
}
ll get(int l, int r){
if (!n) return 0LL;
return get(l, r, 0, n, 0);
}
};
const int N = 1e5 + 42;
vec <pii> g[N];
int us[N], ss[N];
void siz(int u, int p){
// cout << "FUCK " << u << "\n";
ss[u] = 1;
for (auto &[v, id]: g[u]) if (v != p && !us[v]){
siz(v, u), ss[u] += ss[v];
}
}
int _cen(int u, int p, int n){
// cout << "FUCK2 " << u << "\n";
for (auto &[v, id]: g[u]) if (v != p && !us[v]){
if (2 * ss[v] > n){
return _cen(v, u, n);
}
}
return u;
}
int cen(int u){
siz(u, u);
return _cen(u, u, ss[u]);
}
ll w[N];
vec <array <int, 4>> ed[N];
multiset <ll, greater <ll>> val[N], s;
segt t[N];
vec <pii> sub[N];
ll get(int u, int i){
return t[u].get(sub[u][i].fi, sub[u][i].se);
}
ll diam(int u){
ll res = 0;
auto it = val[u].begin();
for (int i = 0; it != val[u].end() && i < 2; ++i, ++it){
res += *it;
}
return res;
}
void upd(int u, int i, int l, int r, ll x){
s.erase(diam(u));
val[u].erase(val[u].find(get(u, i)));
t[u].upd(l, r, x);
val[u].insert(get(u, i));
s.insert(diam(u));
}
int fi[N], fo[N];
int tmr = 0;
void dfs(int u, int p, int c, int pt, ll h, vec <ll> &d){
// cout << "FUCK5 " << u << "\n";
fi[u] = tmr++;
d.push_back(h);
for (auto &[v, id]: g[u]) if (v != p && !us[v]){
dfs(v, u, c, pt, h + w[id], d);
ed[id].push_back({c, pt, fi[v], fo[v]});
}
fo[u] = tmr;
}
void dec(int u){
us[u] = 1;
tmr = 0;
vec <ll> d;
for (auto &[v, id]: g[u]) if (!us[v]){
dfs(v, u, u, sz(sub[u]), w[id], d);
ed[id].push_back({u, sz(sub[u]), fi[v], fo[v]});
sub[u].push_back({fi[v], fo[v]});
}
// cout << "WHERE\n";
t[u] = segt(sz(d));
t[u].build(d);
for (int i = 0; i < sz(sub[u]); ++i){
val[u].insert(get(u, i));
}
s.insert(diam(u));
// cout << "WTF\n";
for (auto &[v, id]: g[u]) if (!us[v]){
dec(cen(v));
}
}
inline int solve(){
int n, q; ll W;
cin >> n >> q >> W;
for (int i = 0; i < n - 1; ++i){
int u, v;
cin >> u >> v >> w[i], --u, --v;
g[u].push_back({v, i});
g[v].push_back({u, i});
}
s = {0LL};
dec(cen(0));
ll ans = 0;
while(q--){
int d; ll e;
cin >> d >> e;
d = (d + ans) % (n - 1), e = (e + ans) % W;
for (auto [c, su, l, r]: ed[d]){
upd(c, su, l, r, e - w[d]);
}
w[d] = e;
cout << (ans = *s.begin()) << "\n";
}
return 0;
}
inline void precalc(){}
signed main(){
ios_base::sync_with_stdio(0); cin.tie(0); cout.tie(0);
int tst = 1; //cin >> tst;
precalc();
while(tst--) solve();
return 0;
}
Compilation message (stderr)
diameter.cpp: In member function 'void segt::build(int, int, int, std::vector<long long int>&)':
diameter.cpp:45:25: warning: suggest parentheses around '+' inside '>>' [-Wparentheses]
45 | int tm = tl + tr >> 1;
| ~~~^~~~
diameter.cpp: In member function 'void segt::upd(int, int, ll, int, int, int)':
diameter.cpp:73:25: warning: suggest parentheses around '+' inside '>>' [-Wparentheses]
73 | int tm = tl + tr >> 1;
| ~~~^~~~
diameter.cpp: In member function 'll segt::get(int, int, int, int, int)':
diameter.cpp:87:25: warning: suggest parentheses around '+' inside '>>' [-Wparentheses]
87 | int tm = tl + tr >> 1;
| ~~~^~~~
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |