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
use std::collections::VecDeque;

/// 根付き木の LCA です。
///
/// # Examples
/// ```
/// use lowest_common_ancestor::LowestCommonAncestor;
///
/// // 0 -- 2 -- 4
/// // |    |
/// // 1    3
///
/// let lca = LowestCommonAncestor::new(5, 0, &[(0, 1), (0, 2), (2, 3), (2, 4)]);
/// assert_eq!(lca.get(0, 1), 0);
/// assert_eq!(lca.get(0, 4), 0);
/// assert_eq!(lca.get(1, 1), 1);
/// assert_eq!(lca.get(1, 2), 0);
/// assert_eq!(lca.get(2, 3), 2);
/// assert_eq!(lca.get(3, 4), 2);
/// ```
pub struct LowestCommonAncestor {
    n: usize,
    ancestor: Vec<Vec<usize>>,
    depth: Vec<usize>,
}

const ILLEGAL: usize = std::usize::MAX;

impl LowestCommonAncestor {
    /// 頂点数 `n`, 根 `root`, 木をなす無向辺の集合 `edges` を渡します。
    pub fn new(n: usize, root: usize, edges: &[(usize, usize)]) -> Self {
        assert!(root < n);
        let mut g = vec![vec![]; n];
        for &(u, v) in edges {
            g[u].push(v);
            g[v].push(u);
        }
        let mut depth = vec![0; n];
        let mut parent = vec![ILLEGAL; n];
        let mut que = VecDeque::new();
        depth[root] = 0;
        que.push_back((root, ILLEGAL));
        while let Some((curr, prev)) = que.pop_front() {
            for &next in &g[curr] {
                if next != prev {
                    depth[next] = depth[curr] + 1;
                    parent[next] = curr;
                    que.push_back((next, curr));
                }
            }
        }
        let table_size = if n == 1 {
            1
        } else {
            // log2(n) の切り上げ
            n.ilog2() as usize + usize::from(!n.is_power_of_two())
        };
        let mut ancestor = vec![vec![ILLEGAL; n]; table_size];
        ancestor[0] = parent;
        for i in 1..table_size {
            ancestor[i] = (0..n)
                .map(|v| {
                    if ancestor[i - 1][v] == ILLEGAL {
                        ILLEGAL
                    } else {
                        ancestor[i - 1][ancestor[i - 1][v]]
                    }
                })
                .collect();
        }
        Self { n, ancestor, depth }
    }

    /// `u` と `v` の LCA を返します。
    pub fn get(&self, u: usize, v: usize) -> usize {
        assert!(u < self.n);
        assert!(v < self.n);
        let (mut u, mut v) = if self.depth[u] >= self.depth[v] {
            (u, v)
        } else {
            (v, u)
        };
        assert!(self.depth[u] >= self.depth[v]);
        let depth_diff = self.depth[u] - self.depth[v];
        for i in 0..self.ancestor.len() {
            if depth_diff >> i & 1 == 1 {
                u = self.ancestor[i][u];
            }
        }
        if u == v {
            return u;
        }
        for i in (0..self.ancestor.len()).rev() {
            if self.ancestor[i][u] != self.ancestor[i][v] {
                u = self.ancestor[i][u];
                v = self.ancestor[i][v];
            }
        }
        let lca = self.ancestor[0][u];
        assert_ne!(lca, ILLEGAL);
        lca
    }

    /// `u` と `v` の距離 (頂点間にある辺の数) を返します。
    pub fn get_dist(&self, u: usize, v: usize) -> usize {
        let w = self.get(u, v);
        self.depth[u] + self.depth[v] - self.depth[w] * 2
    }

    /// 頂点 `u` の深さを返します。
    pub fn depth(&self, u: usize) -> usize {
        self.depth[u]
    }

    /// 頂点 `u` から根の方向に `k` 本の辺を登って着く頂点を返します。
    pub fn kth_parent(&self, u: usize, k: usize) -> Option<usize> {
        assert!(u < self.n);
        if k >= self.n - 1 {
            return None;
        }
        let mut u = u;
        for i in 0..self.ancestor.len() {
            if self.depth[k] >> i & 1 == 1 {
                u = self.ancestor[i][u];
                if u == ILLEGAL {
                    return None;
                }
            }
        }
        Some(u)
    }
}

#[cfg(test)]
mod tests {
    use crate::LowestCommonAncestor;

    #[test]
    fn single_node_test() {
        let lca = LowestCommonAncestor::new(1, 0, &[]);
        assert_eq!(lca.get(0, 0), 0);
    }
}