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
use crate::graph::euler_tour::euler_tour;
use crate::structure::sparse_table::{Band, SparseTable};
struct MinDepth {}
impl Band for MinDepth {
type T = (i32, i32);
fn operate(a: &Self::T, b: &Self::T) -> Self::T {
if a.1 < b.1 {
*a
} else {
*b
}
}
}
pub struct LowestCommonAncestor {
st: SparseTable<MinDepth>,
first_look: Vec<usize>,
}
impl LowestCommonAncestor {
#[inline]
pub fn new(e: &[Vec<usize>], root: usize) -> Self {
let (tour, first_look, depths) = euler_tour(e, root);
let v = tour
.iter()
.map(|&x| (x as i32, depths[x] as i32))
.collect::<Vec<_>>();
let st = SparseTable::new(&v);
LowestCommonAncestor { st, first_look }
}
#[inline]
pub fn get_lca(&self, u: usize, v: usize) -> usize {
if u == v {
return u;
}
let range = if self.first_look[u] < self.first_look[v] {
self.first_look[u]..self.first_look[v]
} else {
self.first_look[v]..self.first_look[u]
};
self.st.fold(range).0 as usize
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_lca() {
let n = 5;
let mut e = vec![vec![]; n];
for (i, &v) in [0, 0, 2, 2].iter().enumerate() {
e[v].push(i + 1);
}
let lca = LowestCommonAncestor::new(&e, 0);
for &(u, v, ans) in [
(0, 0, 0),
(0, 1, 0),
(0, 2, 0),
(0, 3, 0),
(0, 4, 0),
(1, 1, 1),
(1, 2, 0),
(1, 3, 0),
(1, 4, 0),
(2, 2, 2),
(2, 3, 2),
(2, 4, 2),
(3, 3, 3),
(3, 4, 2),
(4, 4, 4),
]
.iter()
{
assert_eq!(lca.get_lca(u, v), ans);
}
}
}