I am writing a function of the following format:
fn pop<T>(data: &mut Vec<Option<T>>) -> Option<T> {
// Let the item be the current element at head
let item = data[0];
// and "remove" it.
data[0] = None;
item
}
When I try to do this, I get an error which makes sense:
error[E0507]: cannot move out of index of `std::vec::Vec<std::option::Option<T>>`
--> src/lib.rs:3:16
|
3 | let item = data[0];
| ^^^^^^^ move occurs because value has type `std::option::Option<T>`, which does not implement the `Copy` trait
|
help: consider borrowing the `Option`'s content
|
3 | let item = data[0].as_ref();
| ^^^^^^^^^^^^^^^^
help: consider borrowing here
|
3 | let item = &data[0];
| ^^^^^^^^
When I try to change it such that item
is a reference, I get an error when I try to set data[0]
to None
, which also makes sense.
Is there some way I can do what I want to do? It seems to me that, whether I want to return a reference or not, I'm going to have to take ownership of the element from the Vec.
I noticed that Vec
has a swap_remove
method, which does almost exactly what I want, except that it swaps with an element already in the Vec
, not with any arbitrary value as I would like. I know that I could just append None
to the end of the Vec
and use swap_remove
, but I'm interested in seeing if there's another way.
Use std::mem::replace
:
use std::mem;
fn pop<T>(data: &mut Vec<Option<T>>) -> Option<T> {
mem::replace(&mut data[0], None)
}
replace
essentially replaces the value in a particular location with another one and returns the previous value.
See also:
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With