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.假设方法中含有泛型参数,编译时无法确认内存大小
相关推荐
武子康28 分钟前
大数据-128 - Flink 并行度详解:从概念到最佳实践,一文读懂任务并行执行机制 代码示例与性能优化
大数据·后端·flink
小毛驴8501 小时前
在Spring Boot开发中,HEAD、OPTIONS和 TRACE这些HTTP方法各有其特定的应用场景和实现方式
spring boot·后端·http
zl9798991 小时前
SpringBoot-依赖管理和自动配置
spring boot·后端·状态模式
JaguarJack1 小时前
PHP8.5 的新 URI 扩展
后端·php
绝无仅有1 小时前
面试真实经历某商银行大厂数据库MYSQL问题和答案总结(一)
后端·面试·github
绝无仅有1 小时前
Docker 实战经验之关键文件误删恢复指南
后端·面试·github
QZQ541882 小时前
go中reflect的底层原理
后端
白衣鸽子2 小时前
CAP理论:分布式系统的“不可能三角”
后端·架构
焰火19992 小时前
[Java]基于Spring的轻量级定时任务动态管理框架
java·后端
Victor3562 小时前
Redis(69)Redis分布式锁的优点和缺点是什么?
后端