Smart pointers … are data structures that act like a pointer but also have additional metadata and capabilities.
Box<T>
Rc<T>
Ref<T>
RefMut<T>
RefCell<T>
Box<T>
Box<T>
Points to data on the heap
Box<T>
UsageT
unkown & need to use value of T
in context where size is neededT
implements a trait, not specific type of T
Box
escons list
Each element contains
Box
esBox
esBox
esBox
esDeref
& Drop
Deref
traitCustomize behavior of dereference *
operator
Code agnostic to pointer type
Box<T>
as a pointerOurBox
OurBox
OurBox
Deref
coercionIf T
implements Deref
, then &T
can be converted to &U
.
Deref
coercionDrop
traitCustomize cleanup behavior when value goes out of scope
Drop
traitstd::mem::drop
Rc<T>
Rc<T>
Rc<T>
Rc<T>
Rc<T>
Rc<T>
fn main() {
let a = Rc::new(Cons(5, Rc::new(Cons(10, Rc::new(Nil)))));
println!("count after creating a = {}", Rc::strong_count(&a));
let b = Cons(3, Rc::clone(&a));
println!("count after creating b = {}", Rc::strong_count(&a));
{
let c = Cons(4, Rc::clone(&a));
println!("count after creating c = {}", Rc::strong_count(&a));
}
println!("count after c goes out of scope = {}", Rc::strong_count(&a));
}
RefCell<T>
RefCell<T>
RefCell<T>
- mock objectsCreate a library to track a value against a maximum value & sends message
Create a library to track a value relative to a maximum value
Send message with status of tracked value
Messenger
traitUse a mock object for testing
RefCell<T>
- mock objectspub struct LimitTracker<'a, T: Messenger> {
messenger: &'a T,
value: usize,
max: usize,
}
impl<'a, T> LimitTracker<'a, T>
where
T: Messenger,
{
pub fn new(messenger: &'a T, max: usize) -> LimitTracker<'a, T> {
LimitTracker {
messenger,
value: 0,
max,
}
}
pub fn set_value(&mut self, value: usize) {
self.value = value;
let percentage_of_max = self.value as f64 / self.max as f64;
if percentage_of_max >= 1.0 {
self.messenger.send("Error: You are over your quota!");
} else if percentage_of_max >= 0.9 {
self.messenger
.send("Urgent warning: You've used up over 90% of your quota!");
} else if percentage_of_max >= 0.75 {
self.messenger
.send("Warning: You've used up over 75% of your quota!");
}
}
}
pub trait Messenger {
fn send(&self, msg: &str);
}
RefCell<T>
- mock objects#[cfg(test)]
mod tests {
use super::*;
struct MockMessenger {
sent_messages: Vec<String>,
}
impl MockMessenger {
fn new() -> MockMessenger {
MockMessenger {
sent_messages: vec![],
}
}
}
impl Messenger for MockMessenger {
fn send(&self, message: &str) {
self.sent_messages.push(String::from(message));
}
}
#[test]
fn it_sends_an_over_75_percent_warning_message() {
let mock_messenger = MockMessenger::new();
let mut limit_tracker = LimitTracker::new(&mock_messenger, 100);
limit_tracker.set_value(80);
assert_eq!(mock_messenger.sent_messages.len(), 1);
}
}
RefCell<T>
- mock objects#[cfg(test)]
mod tests {
use super::*;
use std::cell::RefCell;
struct MockMessenger {
sent_messages: RefCell<Vec<String>>,
}
impl MockMessenger {
fn new() -> MockMessenger {
MockMessenger {
sent_messages: RefCell::new(vec![]),
}
}
}
impl Messenger for MockMessenger {
fn send(&self, message: &str) {
self.sent_messages.borrow_mut().push(String::from(message));
}
}
#[test]
fn it_sends_an_over_75_percent_warning_message() {
// --snip--
assert_eq!(mock_messenger.sent_messages.borrow().len(), 1);
}
}
RefCell<T>
& Rc<T>
RefCell<T>
+ Rc<T>
= mutable data with multiple owners
RefCell<T>
& Rc<T>
Rc<T>
a
Rc<T>
+ RefCell<T>
RefCell<T>
& Rc<T>
#[derive(Debug)]
enum List {
Cons(Rc<RefCell<i32>>, Rc<List>),
Nil,
}
use crate::List::{Cons, Nil};
use std::cell::RefCell;
use std::rc::Rc;
fn main() {
let value = Rc::new(RefCell::new(5));
let a = Rc::new(Cons(Rc::clone(&value), Rc::new(Nil)));
let b = Cons(Rc::new(RefCell::new(3)), Rc::clone(&a));
let c = Cons(Rc::new(RefCell::new(4)), Rc::clone(&a));
*value.borrow_mut() += 10;
println!("a after = {a:?}");
println!("b after = {b:?}");
println!("c after = {c:?}");
}
RefCell<T>
& Rc<T>
- memory leaksRc<T>
with Weak<T>
Weak<T>
Rc<T>
clean upWeak<T>
Weak<T>
When to use Box<T>
, Rc<T>
, and RefCell<T>