Cannot borrow variable as mutable more than once at a time after calling a &'a mut self method

前端 未结 2 1346
耶瑟儿~
耶瑟儿~ 2021-01-22 01:46

I have a problem with lifetimes/borrowing with my Graph object.

fn main() {
    let mut g = Graph {
        nodePointer: &mut 0,
        edgePoi         


        
相关标签:
2条回答
  • 2021-01-22 02:33

    Your problem arises from a misuse of lifetimes, specifically in your signature of add_node:

    pub fn add_node(&'a mut self, data: (u64, u64)) -> usize
    

    In this signature, you are stating that add_node takes an &'a mut self on a Graph<'a>; in other words, you are telling Rust that this method needs to take a mutable borrow on the graph that can't be dropped before the end of the Graph's lifetime, 'a. But since it's the graph itself holding a reference to the graph, the only time that reference will be dropped is when the graph itself is dropped.

    Since add_node doesn't require you to return a reference to any object within the struct, holding onto that borrow is irrelevant. If you alter your add_node method to remove the explicit lifetime:

    pub fn add_node(&mut self, data: (u64, u64)) -> usize
    

    then your example no longer raises an error, because add_node is now only borrowing self until it's finished with the function. (Under the hood, this effectively creates a second lifetime 'b and makes the signature into &'b mut self)

    See the playground for proof.

    0 讨论(0)
  • 2021-01-22 02:33

    Consider this :

    struct Foo<'a> {
        x: &'a i32,
    }
    

    As the book states:

    So why do we need a lifetime here? We need to ensure that any reference to a Foo cannot outlive the reference to an i32 it contains.

    If you write something like:

    impl<'a> Graph<'a> {
        pub fn add_node(&'a mut self, data: (u64, u64)) -> usize {
            ...
    

    the lifetime declaration &'a mut self is not for the purpose of relating the lifetime of Graph instance with the contained references, but for declaring that for mutable self references hold the same lifetime 'a declared for Graph field references:

    fn main() {    
        let mut g = Graph {                          // <------------
            nodePointer: &mut 0,                     //             |            
            edgePointer: &mut 0,                     // lifetime    |
            nodes: &mut Vec::new(),                  // of Graph    | 'a
            edges: &mut Vec::new(),                  // references  |
        };                                           //             |
        let node1 = Graph::add_node(&mut g, (1, 1)); //             |   
        let node2 = Graph::get_node(&mut g, 0);      //             |
    }                                                //<-------------
    

    Where g.get_node(0) has been rewritten as Graph::get_node(&mut g, 0) just for explicitly exposing the &mut reference

    Looking at the lifetime of 'a it is clear that the reference &mut g is borrowed mutably more than once, and this causes the error.

    0 讨论(0)
提交回复
热议问题