Explanation
I'm attempting to implement a binary search tree in Rust. The problem is in the insert
method, where I have a &mut Box<Node<T>>
called node_to_insert_at
. To insert the given value, I travel down the binary search tree, replacing node_to_insert_at
with it's left or right child depending on the value I'm trying to insert's (value
from the function signature) relation to the current node. I'm attempting to store the path to the node that will be inserted in node_to_insert_at_path
which I do by pushing node_to_insert_at
into the stack (Vec) before I overwrite node_to_insert_at
with it's left or right child.
The problem is in the last three lines of this code snippet:
if value < node_to_insert_at.value {
if node_to_insert_at.left.is_some() {
let temp = node_to_insert_at.left.as_mut().unwrap();
node_to_insert_at_path.push(node_to_insert_at);
node_to_insert_at = temp;
Rust says that the line node_to_insert_at_path.push(node_to_insert_at);
creates a second mutable reference, which is illegal. However, I'm unsure how this could be the case, since on the line above it I am only creating a mutable reference to node_to_insert_at
's left value.
Code in Full
For reference, this is what the Node
struct looks like (Node
is in the components file):
pub struct Node<T> {
pub value: T,
pub left: Option<Box< Node<T>>>,
pub right: Option<Box< Node<T>>>
}
Here's the entire file looks like for reference:
use crate::components;
use std::vec::Vec;
use crate::components::Node;
pub struct BSTree<T>{
vars: components::BSTVars<T>
}
impl<T> Default for BSTree<T>{
fn default() -> Self {
BSTree{vars: components::BSTVars{root: None, name: String::from("BSTree"), count: 0}}
}
}
impl<T: Clone + PartialOrd> components::BSTMethods<T> for BSTree<T>{
fn insert(&mut self, value: T){
if self.vars.root.is_none(){
self.vars.root = Option::Some(Box::new(Node {value, left: None, right: None}));
return;
}
let mut node_to_insert_at = self.vars.root.as_mut().unwrap();
let mut node_to_insert_at_path: Vec<&mut Box<Node<T>>> = Vec::new();
let mut not_inserted = true;
while not_inserted {
if value < node_to_insert_at.value {
if node_to_insert_at.left.is_some() {
let temp = node_to_insert_at.left.as_mut().unwrap();
node_to_insert_at_path.push(node_to_insert_at);
node_to_insert_at = temp;
} else {
node_to_insert_at.left = Option::Some(Box::new(Node {value: value.clone(), left: None, right: None}));
not_inserted = false;
}
} else if value > node_to_insert_at.value {
/*if node_to_insert_at.right.is_some() {
node_to_insert_at_path.push(node_to_insert_at);
node_to_insert_at = node_to_insert_at.right.as_mut().unwrap();
} else {
node_to_insert_at.right = Option::Some(Box::new(Node{value: value.clone(), left: None, right: None}))
}*/
} else {
node_to_insert_at.right = Option::Some(Box::new(Node{value: value.clone(), left: None, right: None}));
not_inserted = false;
}
}
}
// impl continues...
}
Possible Explanations & Solutions
From what I gather from this question I might be performing a reborrow, which I think would invalidate node_to_insert_at
because I'm mutably borrowing one of its member variables (please correct me on this if I'm wrong).
To fix this, the stack should be a list of immutable references, which I turn into mutable references as needed by dereferencing the immutable reference.
Is this explanation and solution correct, or is there something I missed?