<style>
.textleft {
text-align:left;
}
.reveal, .reveal h1, .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 {
font-family:Arial, Microsoft JhengHei;}
.reveal .progress {
height: 14px !important;
}
.progress span {
background: url() repeat-x !important;
}
.progress span:after, .progress span.nyancat {
content: "";
background: url() ;
width: 34px !important;
height: 21px !important;
border: none !important;
float: right;
margin-top: -7px;
margin-right: -10px;
}
.my-icon-css {
width: 100%;
height: 200px;
background-position:center;
background-repeat: no-repeat;
background-image:url('https://mir-s3-cdn-cf.behance.net/project_modules/disp/fe36cc42774743.57ee5f329fae6.gif');
}
</style>
## Smart Pointers
#### @wusyong, 吳昱緯
----
### Pointers
- General concept for a variable that contains an address in memory
- Address refers to, or “points at,” some other data
- References are indicated by the & symbol and borrow the value they point to
----
![](https://i.imgur.com/um0g2Kk.png)
----
### Smart Pointers
- Data structures that not only act like a pointer but also have additional metadata and capabilities
- Dynamic allocation
- Modify immutable data
- Own the data they point to
- Usually implemented by struct
- ~~No smart enough~~
----
![](https://i.imgur.com/Jld1P86.png)
----
### Smart Pointers
- Dereference: Deref trait
- Free memory: Drop trait
Smart Pointers = Struct + Deref + Drop
---
## Deref: Treating Smart Pointers Like Regular References
- Customize dereference operator `*`
- DerefMut: dereference of &mut
- Deref coercions: Implicit dereference
----
### Deref trait
```rust
pub trait Deref {
/// The resulting type after dereferencing.
type Target: T;
/// Dereferences the value.
fn deref(&self) -> &Self::Target;
}
```
----
### Following the Pointer to the Value with the Dereference Operator
```rust
fn main() {
let x = 5;
let y = &x;
assert_eq!(5, x);
assert_eq!(5, *y);
}
error[E0277]: the trait bound `{integer}: std::cmp::PartialEq<&{integer}>` is
not satisfied
--> src/main.rs:6:5
|
6 | assert_eq!(5, y);
| ^^^^^^^^^^^^^^^^^ can't compare `{integer}` with `&{integer}`
|
= help: the trait `std::cmp::PartialEq<&{integer}>` is not implemented for
`{integer}`
```
----
### Defining Our Own Smart Pointer
```rust
struct MyBox<T>(T);
impl<T> MyBox<T> {
fn new(x: T) -> MyBox<T> {
MyBox(x)
}
}
fn main() {
let x = 5;
let y = MyBox::new(x);
assert_eq!(5, x);
assert_eq!(5, *y);
}
error[E0614]: type `MyBox<{integer}>` cannot be dereferenced
--> src/main.rs:14:19
|
14 | assert_eq!(5, *y);
|
```
----
### Treating a Type Like a Reference by Implementing the Deref Trait
```rust
use std::ops::Deref;
impl<T> Deref for MyBox<T> {
type Target = T;
fn deref(&self) -> &T {
&self.0
}
}
// *(y.deref()) == *y
```
----
### Implicit Deref Coercions with Functions and Methods
```rust
fn hello(name: &str) {
println!("Hello, {}!", name);
}
fn main() {
let m = MyBox::new(String::from("Rust"));
hello(&m); // hello(&(*m)[..]);
}
```
----
### How Deref Coercion Interacts with Mutability
* From &T to &U when T: Deref<Target=U>
* From &mut T to &mut U when T: DerefMut<Target=U>
* From &mut T to &U when T: Deref<Target=U>
---
## Drop: Running Code on Cleanup
- Destructor
- RAII
----
### Drop trait
```rust
struct CustomSmartPointer {
data: String,
}
impl Drop for CustomSmartPointer {
fn drop(&mut self) {
println!("Dropping CustomSmartPointer with data `{}`!", self.data);
}
}
fn main() {
let c = CustomSmartPointer { data: String::from("my stuff") };
let d = CustomSmartPointer { data: String::from("other stuff") };
println!("CustomSmartPointers created.");
}
CustomSmartPointers created.
Dropping CustomSmartPointer with data `other stuff`!
Dropping CustomSmartPointer with data `my stuff`!
```
----
### Dropping a Value Early with std::mem::drop
Explicit destructor calls are not allowed
```rust
fn main() {
let c = CustomSmartPointer { data: String::from("some data") };
println!("CustomSmartPointer created.");
drop(c); // c.drop() will have double free!
println!("CustomSmartPointer dropped before the end of main.");
}
CustomSmartPointer created.
Dropping CustomSmartPointer with data `some data`!
CustomSmartPointer dropped before the end of main.
```
---
## Box 📦
- Point to Data on the Heap
- Similar to C++ std::unique_ptr
- pointer size on stack is `1 usize`
- keep content ownership
----
![](https://i.imgur.com/QcdM9y7.png)
----
### Using a `Box<T>` to Store Data on the Heap
```rust
fn main() {
let b = Box::new(5);
println!("b = {}", b);
}
```
----
### Enabling Recursive Types with Boxes
Let's build a cons list. A cons list is a data structure that comes from the Lisp programming language and its dialects. In Lisp, the cons function (short for “construct function”) constructs a new pair from its two arguments, which usually are a single value and another pair. These pairs containing pairs form a list.
----
### Computing the Size of a Non-Recursive Type
```rust
enum List {
Cons(i32, List),
Nil,
}
use crate::List::{Cons, Nil};
fn main() {
let list = Cons(1, Cons(2, Cons(3, Nil)));
}
error[E0072]: recursive type `List` has infinite size
--> src/main.rs:1:1
|
1 | enum List {
| ^^^^^^^^^ recursive type has infinite size
2 | Cons(i32, List),
| ----- recursive without indirection
|
= help: insert indirection (e.g., a `Box`, `Rc`, or `&`) at some point to
make `List` representable
```
----
### Computing the Size of a Non-Recursive Type
![](https://i.imgur.com/eFFeWcG.png)
----
### Using `Box<T>` to Get a Recursive Type with a Known Size
```rust
enum List {
Cons(i32, Box<List>),
Nil,
}
use crate::List::{Cons, Nil};
fn main() {
let list = Cons(1,
Box::new(Cons(2,
Box::new(Cons(3,
Box::new(Nil))))));
}
```
----
### Using `Box<T>` to Get a Recursive Type with a Known Size
![](https://i.imgur.com/GG6svi8.png)
---
## Rc ➡️ 📦 ⬅️
- Reference counting
- Similar to C++ std::shared_ptr
- Single thread
- Use Weak to prevent memory leak (This Chapter)
- USe Arc atomic reference counting (Next Chapter)
----
### Using `Rc<T>` to Share Data
![](https://i.imgur.com/QHXdbVq.png)
----
### Using `Rc<T>` to Share Data
Prefer Rc::clone(&a) than a.clone()
```rust
enum List {
Cons(i32, Rc<List>),
Nil,
}
use crate::List::{Cons, Nil};
use std::rc::Rc;
fn main() {
let a = Rc::new(Cons(5, Rc::new(Cons(10, Rc::new(Nil)))));
let b = Cons(3, Rc::clone(&a));
let c = Cons(4, Rc::clone(&a));
}
```
----
### Cloning an `Rc<T>` Increases the Reference Count
```rust
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));
}
count after creating a = 1
count after creating b = 2
count after creating c = 3
count after c goes out of scope = 2
```
---
## Cell and RefCell 📦 💵 🎩 🐇
- Interior mutability
- Cell: Ownership transfer
- RefCell: Enforcing Borrowing Rules at Runtime
- Not on heap
----
### Interior mutability
Uses unsafe code(`UnsafeCell<T>`) inside a data structure to bend Rust’s usual rules that govern mutation and borrowing.
----
![](https://i.imgur.com/t4AdBCE.png)
----
### Cell struct
```rust
pub struct Cell<T: ?Sized> {
value: UnsafeCell<T>,
}
use std::cell::Cell;
let c = Cell::new(5);
c.set(10); // discard 5 and value set to 10
c.get(); // 10 (a copy but not a reference)
c.replace(7) // 10 (returns the replaced value)
```
----
### RefCell struct
```rust
pub struct RefCell<T: ?Sized> {
borrow: Cell<BorrowFlag>,
value: UnsafeCell<T>,
}
```
----
### Enforcing Borrowing Rules at Runtime with `RefCell<T>`
* Recap borrow rules
* At any given time, you can have either (but not both of) one mutable reference or any number of immutable references.
* References must always be valid.
* When you’re sure your code follows the borrowing rules but the compiler is unable to understand and guarantee that
----
### A Mutable Borrow to an Immutable Value
```rust
fn main() {
let x = 5;
let y = &mut x;
}
error[E0596]: cannot borrow immutable local variable `x` as mutable
--> src/main.rs:3:18
|
2 | let x = 5;
| - consider changing this to `mut x`
3 | let y = &mut x;
| ^ cannot borrow mutably
```
----
### A Use Case for Interior Mutability: Mock Objects
```rust
pub trait Messenger {
fn send(&self, msg: &str);
}
pub 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: &T, max: usize) -> LimitTracker<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!");
}
}
}
```
----
### Mocking test
```rust
#[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);
}
}
error[E0596]: cannot borrow immutable field `self.sent_messages` as mutable
--> src/lib.rs:52:13
|
51 | fn send(&self, message: &str) {
| ----- use `&mut self` here to make mutable
52 | self.sent_messages.push(String::from(message));
| ^^^^^^^^^^^^^^^^^^ cannot mutably borrow immutable field
```
----
### RefCell to the rescue
```rust
#[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);
}
}
```
----
### Keeping Track of Borrows at Runtime with `RefCell<T>`
- borrow
- &T
- Return `Ref<T>`
- RefCell::borrow_mut
- &mut T
- Return `RefMut<T>`
----
### Having Multiple Owners of Mutable Data by Combining `Rc<T>` and `RefCell<T>`
```rust
#[derive(Debug)]
enum List {
Cons(Rc<RefCell<i32>>, Rc<List>),
Nil,
}
use crate::List::{Cons, Nil};
use std::rc::Rc;
use std::cell::RefCell;
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(6)), Rc::clone(&a));
let c = Cons(Rc::new(RefCell::new(10)), Rc::clone(&a));
*value.borrow_mut() += 10;
println!("a after = {:?}", a);
println!("b after = {:?}", b);
println!("c after = {:?}", c);
}
a after = Cons(RefCell { value: 15 }, Nil)
b after = Cons(RefCell { value: 6 }, Cons(RefCell { value: 15 }, Nil))
c after = Cons(RefCell { value: 10 }, Cons(RefCell { value: 15 }, Nil))
```
---
## Reference Cycles Can Leak Memory
```rust
use std::rc::Rc;
use std::cell::RefCell;
use crate::List::{Cons, Nil};
#[derive(Debug)]
enum List {
Cons(i32, RefCell<Rc<List>>),
Nil,
}
impl List {
fn tail(&self) -> Option<&RefCell<Rc<List>>> {
match self {
Cons(_, item) => Some(item),
Nil => None,
}
}
}
```
----
### Creating a reference cycle of two List values pointing to each other
```rust
fn main() {
let a = Rc::new(Cons(5, RefCell::new(Rc::new(Nil))));
println!("a initial rc count = {}", Rc::strong_count(&a));
println!("a next item = {:?}", a.tail());
let b = Rc::new(Cons(10, RefCell::new(Rc::clone(&a))));
println!("a rc count after b creation = {}", Rc::strong_count(&a));
println!("b initial rc count = {}", Rc::strong_count(&b));
println!("b next item = {:?}", b.tail());
if let Some(link) = a.tail() {
*link.borrow_mut() = Rc::clone(&b);
}
println!("b rc count after changing a = {}", Rc::strong_count(&b));
println!("a rc count after changing a = {}", Rc::strong_count(&a));
// Uncomment the next line to see that we have a cycle;
// it will overflow the stack
// println!("a next item = {:?}", a.tail());
}
a initial rc count = 1
a next item = Some(RefCell { value: Nil })
a rc count after b creation = 2
b initial rc count = 1
b next item = Some(RefCell { value: Cons(5, RefCell { value: Nil }) })
b rc count after changing a = 2
a rc count after changing a = 2
```
----
![](https://i.imgur.com/ew6ql3a.png)
----
### Preventing Reference Cycles: Turning an `Rc<T>` into a `Weak<T>`
- Weak reference
- Rc::downgrade / Rc::upgrade
- `Rc<T>` instance is only cleaned up if its strong_count is 0
- weak_count doesn’t need to be 0 for the `Rc<T>` instance to be cleaned up.
----
### Creating a Tree Data Structure: a Node with Child Nodes
```rust
use std::rc::Rc;
use std::cell::RefCell;
#[derive(Debug)]
struct Node {
value: i32,
children: RefCell<Vec<Rc<Node>>>,
}
fn main() {
let leaf = Rc::new(Node {
value: 3,
children: RefCell::new(vec![]),
});
let branch = Rc::new(Node {
value: 5,
children: RefCell::new(vec![Rc::clone(&leaf)]),
});
}
```
----
### Adding a Reference from a Child to Its Parent
```rust
use std::rc::{Rc, Weak};
use std::cell::RefCell;
#[derive(Debug)]
struct Node {
value: i32,
parent: RefCell<Weak<Node>>,
children: RefCell<Vec<Rc<Node>>>,
}
fn main() {
let leaf = Rc::new(Node {
value: 3,
parent: RefCell::new(Weak::new()),
children: RefCell::new(vec![]),
});
println!("leaf parent = {:?}", leaf.parent.borrow().upgrade());
// leaf parent = None
let branch = Rc::new(Node {
value: 5,
parent: RefCell::new(Weak::new()),
children: RefCell::new(vec![Rc::clone(&leaf)]),
});
*leaf.parent.borrow_mut() = Rc::downgrade(&branch);
println!("leaf parent = {:?}", leaf.parent.borrow().upgrade());
//leaf parent = Some(Node { value: 5, parent: RefCell { value: (Weak) },
//children: RefCell { value: [Node { value: 3, parent: RefCell { value: (Weak) },
//children: RefCell { value: [] } }] } })
}
```
----
### Visualizing Changes to strong_count and weak_count
```rust
fn main() {
let leaf = Rc::new(Node {
value: 3,
parent: RefCell::new(Weak::new()),
children: RefCell::new(vec![]),
});
println!(
"leaf strong = {}, weak = {}",
Rc::strong_count(&leaf),
Rc::weak_count(&leaf),
); // 1, 0
{
let branch = Rc::new(Node {
value: 5,
parent: RefCell::new(Weak::new()),
children: RefCell::new(vec![Rc::clone(&leaf)]),
});
*leaf.parent.borrow_mut() = Rc::downgrade(&branch);
println!(
"branch strong = {}, weak = {}",
Rc::strong_count(&branch),
Rc::weak_count(&branch),
); // 1, 1
println!(
"leaf strong = {}, weak = {}",
Rc::strong_count(&leaf),
Rc::weak_count(&leaf),
); // 2, 0
}
println!("leaf parent = {:?}", leaf.parent.borrow().upgrade());
println!(
"leaf strong = {}, weak = {}",
Rc::strong_count(&leaf),
Rc::weak_count(&leaf),
); // 1, 0
}
```
---
## Summary
- Smart Pointers
- Box
- Rc
- RefCell
- Weak
----
### Cow
- ~~靠!~~
- Clone on write
```rust
pub enum Cow<'a, B: ?Sized + 'a>
where B: ToOwned
{
/// Borrowed data.
Borrowed(&'a B),
/// Owned data.
Owned(<B as ToOwned>::Owned),
}
```
{"metaMigratedAt":"2023-06-14T22:50:09.075Z","metaMigratedFrom":"YAML","title":"Smart Pointers","breaks":true,"description":"Smart Pointers","contributors":"[{\"id\":\"9913fcec-c953-46f3-8b1c-33170571dfd1\",\"add\":30081,\"del\":6950}]"}