封装(encapsulation):
rust使用pub关键字来控制代码中哪些模块、类型、函数和方法是公开的,默认都是私有的
继承(inheritance):
rust里没有继承。
rust通过定义trait来实现代码共享。trait中也可以默认实现,具体实现时,可以覆盖trait中的默认实现。
多态(polymorphism):
在rust中可以使用泛型来构建不同类型的抽象,并使用trait约束来限制泛型,从而实现多态
trait对象来存储不同类型的值
trait对象被专门用于抽象某些共有行为
trait对象能够指向实现了指定trait的类型实例
创建trait对象:通过选用一种指针,例如&引用或Box<T>智能指针等,并添加dyn关键字与指定相关trait来创建trait对象
trait对象可以被用在泛型或具体类型所处的位置。
Rust类型系统都会在编译时确保出现在相应位置上的值实现trait对象指定的trait
trait对象VS泛型:
泛型参数一次只能被替代为一个具体的类型,而trait对象则允许你在运行时填入多种不同的具体类型
pub struct Screen {
pub components: Vec<Box<dyn Draw>>,
}
这一概念与动态类型语言中的“鸭子类型”(ducktyping)十分相似:如果某个东西走起来像鸭子,叫起来也像鸭子,那么它就是一只鸭子!
动态数组components时指定Box<dyn Draw>元素类型,Screen实例只会接收那些能够调用draw方法的值。
trait对象会执行动态派发
静态派发(static dispatch):
在泛型使用trait约束时执行单态化,编译器会为每一个具体类型生成对应泛型函数和泛型方法的非泛型实现,并使用这些具体的类型来替换泛型参数
动态派发(dynamic dispatch):
动态派发下的编译器无法在编译过程中确定你调用的究竟是哪一个方法。Rust必然会在我们使用trait对象时执行动态派发。因为编译器无法知晓所有能够用于trait对象的具体类型,所以它无法在编译时确定需要调用哪个类型的哪个具体方法。
trait对象必须保证对象安全
Rust采用了一套较为复杂的规则来决定某个trait是否对象安全:
方法的返回类型不是Self。
方法中不包含任何泛型参数。
关键字Self是一个别名,它指向了实现当前trait或方法的具体类型