-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathcco13p3.cpp
79 lines (79 loc) · 2.13 KB
/
cco13p3.cpp
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
// Ivan Carvalho
// Solution to https://dmoj.ca/problem/cco13p3
#include <algorithm>
#include <cstdio>
#include <map>
#include <vector>
#define MAXN 400010
using namespace std;
typedef map<int, int>::iterator ite;
vector<int> grafo[MAXN];
map<int, int> *conjunto[MAXN];
int aux, maioral, n, tam[MAXN], diametro, nivel[MAXN], ac[MAXN];
long long resp;
void dfs_diametro(int v, int p, int depth) {
if (depth > maioral) {
maioral = depth;
aux = v;
}
for (int u : grafo[v]) {
if (u == p) continue;
dfs_diametro(u, v, depth + 1);
}
}
void dfs_sack(int v, int p) {
tam[v] = 1;
int mx = -1, big = -1;
for (int u : grafo[v]) {
if (u == p) continue;
dfs_sack(u, v);
tam[v] += tam[u];
if (tam[u] > mx) {
mx = tam[u];
big = u;
}
}
if (big == -1) {
conjunto[v] = new map<int, int>();
(*conjunto[v])[0] = 1;
return;
} else {
ac[v] = ac[big] + 1;
conjunto[v] = conjunto[big];
if ((*conjunto[v]).count(diametro - ac[v]))
resp += 1LL * (*conjunto[v])[diametro - ac[v]];
(*conjunto[v])[-ac[v]] = 1;
}
for (int u : grafo[v]) {
if (u == p || u == big) continue;
for (ite it = conjunto[u]->begin(); it != conjunto[u]->end(); it++) {
int dist = (*it).first + 1 + ac[u];
int qtd = (*it).second;
if ((*conjunto[v]).count(diametro - dist - ac[v]))
resp += 1LL * qtd * (*conjunto[v])[diametro - dist - ac[v]];
}
for (ite it = conjunto[u]->begin(); it != conjunto[u]->end(); it++) {
int dist = (*it).first + 1 + ac[u];
int qtd = (*it).second;
(*conjunto[v])[dist - ac[v]] += qtd;
}
}
}
int main() {
scanf("%d", &n);
for (int i = 1; i < n; i++) {
int u, v;
scanf("%d %d", &u, &v);
grafo[u].push_back(v);
grafo[v].push_back(u);
}
dfs_diametro(1, -1, 0);
maioral = 0;
dfs_diametro(aux, -1, 0);
diametro = maioral;
dfs_sack(1, -1);
printf("%d %lld\n", diametro + 1, resp);
return 0;
}
urn 0;
}