Submission #53756


ソースコード

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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
#include "bits/stdc++.h"
#define rep(i, n) for (i64 i = 0, i##_limit = (n); i < i##_limit; ++i)
using namespace std;
using i64 = int_fast64_t;
template<class T, class U>inline bool chmax(T &a, const U &b){return b>a && (a=b, true);}
template<class T, class U>inline bool chmin(T &a, const U &b){return b<a && (a=b, true);}
constexpr int INF = 0x3f3f3f3f;
constexpr i64 LINF = 0x3f3f3f3f3f3f3f3fLL;
struct Edge { // {{{
int src, to;
int64_t cost = 0;
inline Edge(int src, int to, const int64_t &cost) noexcept :
src(src), to(to), cost(cost) {}
inline Edge(int to, const int64_t &cost) noexcept :
src(-1), to(to), cost(cost) {}
inline Edge() noexcept {}
bool operator< (const Edge &o) const noexcept { return cost < o.cost; }
bool operator> (const Edge &o) const noexcept { return o < *this; }
Edge& operator=(int to) noexcept {
this->to = to;
return *this;
}
operator int() const noexcept { return to; }
};
// }}}
using WeightedGraph = vector<vector<Edge>>;
using UnWeightedGraph = vector<vector<int>>;
class HeavyLightDecomposition { // {{{
public:
WeightedGraph G; // 重み付きグラフ
const int V; // ノード数
vector<int> parent; // 親
vector<int> treeSize; // 自分も含めた部分木のサイズ
vector<int> head; // HL分解後の自分が属するグループ(チェイン)の先頭ノード
vector<int> vid; // 根からDFSしたときにそのノードを訪れた順番
vector<int> nodeOfVid; // vidからノードを逆算する用
vector<int64_t> weightSum; // 根からのコストの総和
vector<int> depth; // 根からの深さ(経由するエッジの数)
HeavyLightDecomposition() = delete;
// 空の木を構築する。管理できるノードは [0, V) なので注意。
explicit HeavyLightDecomposition(int V) :
G(V), V(V),
parent(V, -1), treeSize(V), head(V),
vid(V), nodeOfVid(V),
weightSum(V, 0),
depth(V, 0) {}
// u-v間に無向辺を張る。重みは省略可
inline void addEdge(int u, int v, int64_t weight = 1) {
G[u].emplace_back(u, v, weight);
G[v].emplace_back(v, u, weight);
}
// HL分解、その他もろもろ前計算
void build(int root = 0) {
int timestamp = 0;
head[root] = root;
dfs_treeSize(root);
dfs_HLDecomp(root, timestamp);
}
// u,v の最小共通祖先を求める
int lca(int u, int v) const {
while(head[u] != head[v]) {
if (vid[u] > vid[v]) swap(u, v);
v = parent[head[v]];
}
return (vid[u] < vid[v]) ? u : v;
}
// 根に向かって v から k だけ登ったノードを求める
int climb(int v, int k) const {
while(1) {
const int h = head[v];
if (vid[v] - k >= vid[h]) return nodeOfVid[vid[v] - k];
k -= vid[v] - vid[h] + 1;
v = parent[h];
}
}
// u-v間のパスの重みの和を求める
int64_t pathWeight(int u, int v) const {
return weightSum[u] + weightSum[v] - (2 * weightSum[lca(u, v)]);
}
// u-v間のパスの「エッジの個数」を求める
int pathCountEdge(int u, int v) const {
return depth[u] + depth[v] - (2 * depth[lca(u, v)]) + 1;
}
private: // {{{
// treeSize[v], parent[v], depth[v], weightSum[v] を再帰によって計算
// v の子の中で最もheavyなやつを G[v] の先頭にもってくる
int dfs_treeSize(int v) {
treeSize[v] = 1;
if (!G[v].empty() && G[v].front().to == parent[v]) swap(G[v].front(), G[v].back());
for (auto &e : G[v]) {
if (e.to == parent[v]) continue;
parent[e.to] = v;
depth[e.to] = depth[v] + 1;
weightSum[e.to] = weightSum[v] + e.cost;
treeSize[v] += dfs_treeSize(e.to);
if (treeSize[e.to] > treeSize[G[v].front()]) swap(e, G[v].front());
}
return treeSize[v];
}
// G[v] の先頭に v の子中で最もheavyなノードがある前提でHL分解し、
// vid[v], nodeOfVid[v], head[v] を求める
void dfs_HLDecomp(int v, int ×tamp) {
vid[v] = timestamp;
nodeOfVid[timestamp] = v;
++timestamp;
for (const int to : G[v]) {
if (to == parent[v]) continue;
head[to] = (to == G[v].front() ? head[v] : to);
dfs_HLDecomp(to, timestamp);
}
} // }}}
}; // }}}
signed main()
{
ios::sync_with_stdio(false); cin.tie(nullptr);
int N, Q;
cin >> N >> Q;
HeavyLightDecomposition G(N+1);
rep(i, N - 1) {
int u, v, w;
cin >> u >> v >> w;
G.addEdge(u, v, w);
}
// 根のノードを1としてHL分解
G.build(1);
// a-b, a-c, b-c の3つのパスの中で重みが最大のものを探し、
// { パスの始点, パスの終点 } のペアを返す。
const auto maxWeightPath = [&G](int a, int b, int c) -> pair<int,int> {
const auto weightAB = G.pathWeight(a, b);
const auto weightAC = G.pathWeight(a, c);
const auto weightBC = G.pathWeight(b, c);
const auto maxv = max({ weightAB, weightAC, weightBC });
if (maxv == weightAB) return { a, b };
if (maxv == weightAC) return { a, c };
return { b, c };
};
// s からlcaに向かって登る距離を [low, high) で2分探索
// (パス[s-lca]上で、weight(s, p) < weight(t, p) を満たし、s からできるだけ離れたノードを探索)
// 探索途中のノードも解の候補に含めるので逐次 res を更新する
const auto binSearch = [&G](int s, int lca, int t) -> i64 {
i64 res = G.pathWeight(t, s);
int low = 0;
int high = G.pathCountEdge(s, lca) + 1;
while(high - low > 1) {
const int mid = (low + high) / 2;
const int p = G.climb(s, mid);
const auto weightSP = G.pathWeight(s, p);
const auto weightTP = G.pathWeight(t, p);
chmin(res, max(weightSP, weightTP));
if (weightSP < weightTP) {
low = mid;
} else {
high = mid;
}
}
return res;
};
const auto solve = [&](int a, int b, int c) -> i64 {
int s, t;
tie(s, t) = maxWeightPath(a, b, c);
const int lca = G.lca(s, t);
return min(binSearch(s, lca, t), binSearch(t, lca, s));
};
rep(i, Q) {
int a, b, c;
cin >> a >> b >> c;
cout << solve(a, b, c) << "\n";
}
return 0;
}

ステータス

項目 データ
問題 1024 - 待ち合わせ
ユーザー名 Arumakan_ei1727
投稿日時 2019-08-28 22:03:42
言語 C++14
状態 Accepted
得点 14
ソースコード長 6965 Byte
最大実行時間 132 ms
最大メモリ使用量 25528 KB

セット

セット 得点 Cases
1 ALL 14 / 14 *

テストケース

ファイル名 状態 実行時間 メモリ使用量 #
in1.txt AC 27 ms 604 KB
1
in2.txt AC 21 ms 560 KB
1
in3.txt AC 18 ms 668 KB
1
in4.txt AC 18 ms 2672 KB
1
in5.txt AC 18 ms 7320 KB
1
in6.txt AC 24 ms 12372 KB
1
in7.txt AC 18 ms 12324 KB
1
in8.txt AC 21 ms 12536 KB
1
in9.txt AC 19 ms 12356 KB
1
in10.txt AC 30 ms 560 KB
1
in11.txt AC 14 ms 380 KB
1
in12.txt AC 22 ms 588 KB
1
in13.txt AC 24 ms 540 KB
1
in14.txt AC 28 ms 492 KB
1
in15.txt AC 23 ms 568 KB
1
in16.txt AC 20 ms 512 KB
1
in17.txt AC 19 ms 584 KB
1
in18.txt AC 17 ms 400 KB
1
in19.txt AC 20 ms 608 KB
1
in20.txt AC 25 ms 516 KB
1
in21.txt AC 20 ms 468 KB
1
in22.txt AC 29 ms 664 KB
1
in23.txt AC 23 ms 480 KB
1
in24.txt AC 18 ms 416 KB
1
in25.txt AC 17 ms 616 KB
1
in26.txt AC 25 ms 684 KB
1
in27.txt AC 27 ms 616 KB
1
in28.txt AC 23 ms 672 KB
1
in29.txt AC 19 ms 604 KB
1
in30.txt AC 24 ms 624 KB
1
in31.txt AC 26 ms 684 KB
1
in32.txt AC 24 ms 616 KB
1
in33.txt AC 19 ms 552 KB
1
in34.txt AC 21 ms 616 KB
1
in35.txt AC 19 ms 676 KB
1
in36.txt AC 112 ms 19168 KB
1
in37.txt AC 128 ms 11636 KB
1
in38.txt AC 132 ms 10940 KB
1
in39.txt AC 132 ms 13108 KB
1
in40.txt AC 130 ms 12092 KB
1
in41.txt AC 119 ms 12916 KB
1
in42.txt AC 71 ms 13284 KB
1
in43.txt AC 83 ms 12608 KB
1
in44.txt AC 85 ms 12956 KB
1
in45.txt AC 95 ms 15384 KB
1
in46.txt AC 103 ms 15652 KB
1
in47.txt AC 119 ms 15516 KB
1
in48.txt AC 128 ms 17020 KB
1
in49.txt AC 119 ms 17152 KB
1
in50.txt AC 122 ms 17520 KB
1
in51.txt AC 126 ms 18868 KB
1
in52.txt AC 119 ms 22416 KB
1
in53.txt AC 114 ms 23840 KB
1
in54.txt AC 121 ms 23460 KB
1
in55.txt AC 121 ms 24928 KB
1
in56.txt AC 116 ms 25528 KB
1