Can I reassign a mutable slice reference to a sub-slice of itself?

后端 未结 2 409
眼角桃花
眼角桃花 2020-12-07 02:25

I\'m implementing a stack-like structure, where the structure holds a mutable reference to a slice.

struct StackLike<\'a, X> {
    data: &\'a mut          


        
相关标签:
2条回答
  • For sub-question 2, you need to indicate the relation between &mut self and 'a otherwise they're considered unrelated. I don't know if there's a shortcut via lifetime elision but if you specify that self lives for 'a you're fine.

    For sub-question 1, the compiler doesn't "see through" function calls (including indexing which desugars to a function call), so it does not know that &self.data[n - 1] and &mut self.data[0..n-1] are non-overlapping. You need to use split_mut_last.

    struct StackLike<'a, X> {
        data: &'a mut [X],
    }
    
    impl<'a, X> StackLike<'a, X> {
        pub fn pop(&'a mut self) -> Option<&'a X> {
            if let Some((last, subslice)) = self.data.split_last_mut() {
                self.data = subslice;
                Some(last)
            } else {
                None
            }
        }
    }
    

    playground

    0 讨论(0)
  • 2020-12-07 03:20

    I modified Masklinn's code slightly to allow multiple .pop()s to be called on the same stack:

    struct StackLike<'a, X> {
        data: &'a mut [X],
    }
    
    impl<'a, X> StackLike<'a, X> {
        pub fn pop(&mut self) -> Option<&'a mut X> {
            let data = std::mem::replace(&mut self.data, &mut []);
            if let Some((last, subslice)) = data.split_last_mut() {
                self.data = subslice;
                Some(last)
            } else {
                None
            }
        }
    }
    
    fn main() {
        let mut data = [1, 2, 3, 4, 5];
        let mut stack = StackLike { data: &mut data };
    
        let x = stack.pop().unwrap();
        let y = stack.pop().unwrap();
        println!("X: {}, Y: {}", x, y);
    }
    

    The tricky part here is this line (I added a type annotation for explicitness):

    let data: &'a mut [X] = std::mem::replace(&mut self.data, &mut []);
    

    We replace self.data with an empty slice temporarily so that we can split the slice. If you were to write simply

    let data: &'a mut [X] = self.data;
    

    the compiler would be unhappy:

    error[E0312]: lifetime of reference outlives lifetime of borrowed content...
      --> src/main.rs:7:33
       |
    7  |         let data: &'a mut [X] = self.data;
       |                                 ^^^^^^^^^
       |
    note: ...the reference is valid for the lifetime `'a` as defined on the impl at 5:6...
      --> src/main.rs:5:6
       |
    5  | impl<'a,  X> StackLike<'a, X> {
       |      ^^
    note: ...but the borrowed content is only valid for the anonymous lifetime #1 defined on the method body at 6:5
      --> src/main.rs:6:5
       |
    6  | /     pub fn pop(&mut self) -> Option<&'a mut X> {
    7  | |         let data: &'a mut [X] = self.data;
    8  | |         if let Some((last, subslice)) = data.split_last_mut() {
    9  | |             self.data = subslice;
    ...  |
    13 | |         }
    14 | |     }
       | |_____^
    

    As far as I understand it, the problem is that self.data is a mutable reference, and mutable references are not Copy (remember, you can only have one at a time). And you cannot move out of self.data since self is a mutable reference, not an owner. So what the compiler tries to do is to reborrow self.data, which "infects" it with with the lifetime of &mut self. This is a dead end: we want the reference to live for 'a, but it is actually valid only for the lifetime of &mut self, and these lifetimes are generally unrelated (and they don't need to be related), which leaves the compiler perplexed.

    To aid the compiler, we use std::mem::replace to explicitly move the slice out of self.data and temporarily replace it with an empty slice, which can be any lifetime. Now we can do anything with data without tangling it with the lifetime of &mut self.

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