From 0cfa232bcec218aa1051c2dc5d027a6b4c053990 Mon Sep 17 00:00:00 2001 From: bsbds <69835502+bsbds@users.noreply.github.com> Date: Tue, 12 Mar 2024 16:37:58 +0800 Subject: [PATCH] refactor: replace max value with index to avoid cloning Signed-off-by: bsbds <69835502+bsbds@users.noreply.github.com> --- crates/utils/src/interval_map/mod.rs | 118 ++++++++++++------------- crates/utils/src/interval_map/tests.rs | 2 +- 2 files changed, 59 insertions(+), 61 deletions(-) diff --git a/crates/utils/src/interval_map/mod.rs b/crates/utils/src/interval_map/mod.rs index 3b8fea76d..030eaea13 100644 --- a/crates/utils/src/interval_map/mod.rs +++ b/crates/utils/src/interval_map/mod.rs @@ -16,7 +16,7 @@ pub struct IntervalMap { impl IntervalMap where - T: Clone + Ord, + T: Ord, Ix: IndexType, { /// Creates a new `IntervalMap` with estimated capacity. @@ -39,8 +39,8 @@ where /// This method panics when the tree is at the maximum number of nodes for its index #[inline] pub fn insert(&mut self, interval: Interval, value: V) -> Option { - let node = Self::new_node(interval, value); let node_idx = NodeIndex::new(self.nodes.len()); + let node = Self::new_node(interval, value, node_idx); // check for max capacity, except if we use usize assert!( ::max().index() == !0 || NodeIndex::end() != node_idx, @@ -56,9 +56,7 @@ where pub fn remove(&mut self, interval: &Interval) -> Option { if let Some(node_idx) = self.search_exact(interval) { self.remove_inner(node_idx); - // To achieve an O(1) time complexity for node removal, we swap the node - // with the last node stored in the vector and update parent/left/right - // nodes of the last node. + // Swap the node with the last node stored in the vector and update indices let mut node = self.nodes.swap_remove(node_idx.index()); let old = NodeIndex::::new(self.nodes.len()); self.update_idx(old, node_idx); @@ -150,7 +148,7 @@ where impl IntervalMap where - T: Clone + Ord, + T: Ord, { /// Creates an empty `IntervalMap` #[must_use] @@ -166,7 +164,7 @@ where impl Default for IntervalMap where - T: Clone + Ord, + T: Ord, { #[inline] fn default() -> Self { @@ -176,7 +174,7 @@ where impl IntervalMap where - T: Clone + Ord, + T: Ord, Ix: IndexType, { /// Creates a new sentinel node @@ -184,7 +182,7 @@ where Node { interval: None, value: None, - max: None, + max_index: None, left: None, right: None, parent: None, @@ -193,9 +191,9 @@ where } /// Creates a new tree node - fn new_node(interval: Interval, value: V) -> Node { + fn new_node(interval: Interval, value: V, index: NodeIndex) -> Node { Node { - max: Some(interval.high.clone()), + max_index: Some(index), interval: Some(interval), value: Some(value), left: Some(Self::sentinel()), @@ -213,7 +211,7 @@ where impl IntervalMap where - T: Ord + Clone, + T: Ord, Ix: IndexType, { /// Inserts a node into the tree. @@ -300,16 +298,12 @@ where if self.node_ref(x, Node::interval).overlap(interval) { list.push(self.node_ref(x, |nx| (nx.interval(), nx.value()))); } - if self - .left_ref(x, Node::sentinel) - .map(Node::max) - .is_some_and(|lm| lm >= &interval.low) - { + if self.max(self.node_ref(x, Node::left)) >= Some(&interval.low) { list.extend(self.find_all_overlap_inner(self.node_ref(x, Node::left), interval)); } if self - .right_ref(x, Node::sentinel) - .map(|r| IntervalRef::new(&self.node_ref(x, Node::interval).low, r.max())) + .max(self.node_ref(x, Node::right)) + .map(|rmax| IntervalRef::new(&self.node_ref(x, Node::interval).low, rmax)) .is_some_and(|i| i.overlap(interval)) { list.extend(self.find_all_overlap_inner(self.node_ref(x, Node::right), interval)); @@ -325,11 +319,7 @@ where .map(Node::interval) .is_some_and(|xi| !xi.overlap(interval)) { - if self - .left_ref(x, Node::sentinel) - .map(Node::max) - .is_some_and(|lm| lm > &interval.low) - { + if self.max(self.node_ref(x, Node::left)) > Some(&interval.low) { x = self.node_ref(x, Node::left); } else { x = self.node_ref(x, Node::right); @@ -345,7 +335,7 @@ where if self.node_ref(x, Node::interval) == interval { return Some(x); } - if self.node_ref(x, Node::max) < &interval.high { + if self.max(x) < Some(&interval.high) { return None; } if self.node_ref(x, Node::interval) > interval { @@ -512,36 +502,35 @@ where /// Updates the max value after a rotation. fn rotate_update_max(&mut self, x: NodeIndex, y: NodeIndex) { - self.node_mut(y, Node::set_max(self.node_ref(x, Node::max_owned))); - let mut max = &self.node_ref(x, Node::interval).high; - if let Some(lmax) = self.left_ref(x, Node::sentinel).map(Node::max) { - max = max.max(lmax); - } - if let Some(rmax) = self.right_ref(x, Node::sentinel).map(Node::max) { - max = max.max(rmax); - } - self.node_mut(x, Node::set_max(max.clone())); + self.node_mut(y, Node::set_max_index(self.node_ref(x, Node::max_index))); + self.recaculate_max(x); } /// Updates the max value towards the root fn update_max_bottom_up(&mut self, x: NodeIndex) { let mut p = x; while !self.node_ref(p, Node::is_sentinel) { - self.node_mut( - p, - Node::set_max(self.node_ref(p, Node::interval).high.clone()), - ); - self.max_from(p, self.node_ref(p, Node::left)); - self.max_from(p, self.node_ref(p, Node::right)); + self.recaculate_max(p); p = self.node_ref(p, Node::parent); } } - /// Updates a nodes value from a child node. - fn max_from(&mut self, x: NodeIndex, c: NodeIndex) { - if let Some(cmax) = self.node_ref(c, Node::sentinel).map(Node::max) { - let max = self.node_ref(x, Node::max).max(cmax).clone(); - self.node_mut(x, Node::set_max(max)); + /// Recaculate max value from left and right childrens + fn recaculate_max(&mut self, x: NodeIndex) { + self.node_mut(x, Node::set_max_index(x)); + let x_left = self.node_ref(x, Node::left); + let x_right = self.node_ref(x, Node::right); + if self.max(x_left) > self.max(x) { + self.node_mut( + x, + Node::set_max_index(self.node_ref(x_left, Node::max_index)), + ); + } + if self.max(x_right) > self.max(x) { + self.node_mut( + x, + Node::set_max_index(self.node_ref(x_right, Node::max_index)), + ); } } @@ -578,7 +567,10 @@ where self.parent_ref(node, Node::right) == node } - /// Updates nodes index after remove + /// Updates nodes indices after remove + /// + /// This method has a time complexity of `O(logn)`, as we need to + /// update the max index from bottom to top. fn update_idx(&mut self, old: NodeIndex, new: NodeIndex) { if self.root == old { self.root = new; @@ -593,6 +585,14 @@ where } self.left_mut(new, Node::set_parent(new)); self.right_mut(new, Node::set_parent(new)); + + let mut p = new; + while !self.node_ref(p, Node::is_sentinel) { + if self.node_ref(p, Node::max_index) == old { + self.node_mut(p, Node::set_max_index(new)); + } + p = self.node_ref(p, Node::parent); + } } } } @@ -693,6 +693,11 @@ where let grand_parent_idx = self.nodes[parent_idx].parent().index(); op(&mut self.nodes[grand_parent_idx]) } + + fn max(&self, node: NodeIndex) -> Option<&T> { + let max_index = self.nodes[node.index()].max_index?.index(); + self.nodes[max_index].interval.as_ref().map(|i| &i.high) + } } /// An iterator over the entries of a `IntervalMap`. @@ -781,7 +786,7 @@ pub struct VacantEntry<'a, T, V, Ix> { impl<'a, T, V, Ix> Entry<'a, T, V, Ix> where - T: Ord + Clone, + T: Ord, Ix: IndexType, { /// Ensures a value is in the entry by inserting the default if empty, and returns @@ -835,8 +840,8 @@ pub struct Node { /// Interval of the node interval: Option>, - /// Max value of the sub-tree of the node - max: Option, + /// The index that point to the node with the max value + max_index: Option>, /// Value of the node value: Option, } @@ -857,15 +862,8 @@ where self.interval.as_ref().unwrap() } - fn max(&self) -> &T { - self.max.as_ref().unwrap() - } - - fn max_owned(&self) -> T - where - T: Clone, - { - self.max().clone() + fn max_index(&self) -> NodeIndex { + self.max_index.unwrap() } fn left(&self) -> NodeIndex { @@ -918,9 +916,9 @@ where } } - fn set_max(max: T) -> impl FnOnce(&mut Node) { + fn set_max_index(max_index: NodeIndex) -> impl FnOnce(&mut Node) { move |node: &mut Node| { - let _ignore = node.max.replace(max); + let _ignore = node.max_index.replace(max_index); } } diff --git a/crates/utils/src/interval_map/tests.rs b/crates/utils/src/interval_map/tests.rs index 35a624647..4f3e8a018 100644 --- a/crates/utils/src/interval_map/tests.rs +++ b/crates/utils/src/interval_map/tests.rs @@ -56,7 +56,7 @@ impl IntervalMap { let l_max = self.check_max_inner(self.node_ref(x, Node::left)); let r_max = self.check_max_inner(self.node_ref(x, Node::right)); let max = self.node_ref(x, |x| x.interval().high.max(l_max).max(r_max)); - assert_eq!(self.node_ref(x, Node::max_owned), max); + assert_eq!(self.max(x), Some(&max)); max }