Update chapter_3_16_2.md

This commit is contained in:
令狐一冲
2023-05-18 09:43:37 +08:00
committed by GitHub
parent 0a7e71bc0a
commit a994c66582

View File

@@ -22,7 +22,7 @@ fn main() {
- 当有大量数据并希望在确保数据不被拷贝的情况下转移所有权的时候;
- 当希望拥有一个值并只关心它的类型是否实现了特定```trait```而不是其具体类型时。
1场景1示例
### 1场景1示例
假定我们需要采用递归的方式定义一个```List```,其定义可能如下:
```Rust
// 下面的代码无法编译通过
@@ -41,4 +41,78 @@ fn main() {
}
```
但是上面的代码无法编译通过,因为```Cons```类型在编译时无法确定其具体大小。其内存示意图如下:
![注释](../../assets/22.png)
此时就需要使用Box其代码如下
```Rust
use crate::List::{Nil, Cons};
enum List {
Cons(i32, Box<List>), // 用Box就把它变成了一个指针Cons就类似于c语言的结构体定义
// struct List{
// int data;
// struct List *next; //指向的一个指针,指针的大小是固定的
// }
Nil,
}
fn main() {
let _list = Cons(1, Box::new(Cons(2, Box::new(Cons(3, Box::new(Nil))))));
}
```
使用Box后上面的Cons的内存表示如下
![注释](../../assets/21.png)
每个Box的大小是固定的所以编译不会有问题。
### 2场景2示例
```Rust
fn main() {
let b = Box::new([100u32; 100]);
println!("b = {:?}", b);
let c = b;
// println!("b = {:?}", b); // 此行打开将报错因为所有权已经转移到c
println!("c = {:?}", c);
}
```
### 3场景3示例
```Rust
trait Vehicle {
fn run(&self);
}
struct Car(u32);
impl Vehicle for Car {
fn run(&self) {
println!("Car {:?} run ... ", self.0);
}
}
struct Truck(u32);
impl Vehicle for Truck {
fn run(&self) {
println!("Truck {:?} run ... ", self.0);
}
}
//vehicle_run方法的参数要求是一个Vehicle trait对象
fn vehicle_run(vehicle: Box<dyn Vehicle>) {
vehicle.run();
}
fn main() {
let car = Car(1001);
let truck = Truck(1002);
let v1 = Box::new(car);
vehicle_run(v1);
let v2 = Box::new(truck);
vehicle_run(v2);
}
```