I\'m implementing a stack-like structure, where the structure holds a mutable reference to a slice.
struct StackLike<\'a, X> {
data: &\'a mut
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
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
.