Rust中的动态分发

动态分发在Rust中主要通过Trait对象实现,本质是一个胖指针:

  1. 数据指针:指向实际对象的内存地址
  2. 虚表指针

在Rust中,对于结构体和枚举类型而言,其字段的数据与impl块中实现的行为是分离的;而trait对象则无法添加数据,专门用于抽象某些共有行为。

trait对象可以被用在泛型或具体类型所处的位置,Rust类型系统会在编译期确保出现在相应位置上的值实现trait对象指定的trait

trait对象实现的动态数组结构体与带trait约束的泛型参数实现的结构体相异之处在于泛型参数一次只能被替代为一个具体的类型,trait对象则运行运行时填入多种不同的具体类型

差异 trait对象 泛型约束
编译机制 动态分发(运行时虚表查询) 静态分发(编译时单态化生成代码)
内存布局 胖指针存储(数据指针 + 虚表) 连续内存存储
安全要求 trait需要满足对象安全规则 无特殊要求
性能开销 虚表查询开销 无额外开销

使用trait对象与类型系统实现"鸭子类型",无需在运行时检查某个值是否实现了指定的方法或调用了未定义方法,Rust不允许此类代码通过编译。

确保trait对象满足对象安全规则,即trait中定义的方法满足以下2个条件:

  1. 方法的返回类型不是Self
  2. 方法中不含任何泛型参数

如何去思考命题:"trait对象必须是安全的" ?

  1. 什么情况下提出这个条件? 答:使用trait对象与类型系统通过动态分发实现"鸭子类型"
  2. 满足何种规则后trait对象是安全的?答:方法返回类型不是Self && 方法中不含任何泛型参数
  3. 假设规则存在不满足会导致什么后果?答:1.假设返回类型为Self,关键字Self为别名,指向实现当前trait或方法的具体类型,但trait对象需要运行时进行虚表查询以确认具体类型。2.假设方法中含有泛型参数,编译时无法确认内存大小
相关推荐
Coder_Boy_几秒前
基于SpringAI的在线考试系统-考试系统开发流程案例
java·数据库·人工智能·spring boot·后端
掘金者阿豪1 小时前
关系数据库迁移的“暗礁”:金仓数据库如何规避数据完整性与一致性风险
后端
ServBay1 小时前
一个下午,一台电脑,终结你 90% 的 Symfony 重复劳动
后端·php·symfony
sino爱学习1 小时前
高性能线程池实践:Dubbo EagerThreadPool 设计与应用
java·后端
颜酱2 小时前
从二叉树到衍生结构:5种高频树结构原理+解析
javascript·后端·算法
掘金者阿豪2 小时前
UUID的隐形成本:一个让数据库“慢下来”的陷阱
后端
用户084465256372 小时前
Docker 部署 MongoDB Atlas 到服务端
后端
Anita_Sun3 小时前
一看就懂的 Haskell 教程 - 类型推断机制
后端·haskell
Anita_Sun3 小时前
一看就懂的 Haskell 教程 - 类型签名
后端·haskell
七八星天3 小时前
C#代码设计与设计模式
后端