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.假设方法中含有泛型参数,编译时无法确认内存大小
相关推荐
tsumikistep9 分钟前
【前后端】接口文档与导入
前端·后端·python·硬件架构
码事漫谈1 小时前
为什么C语言拒绝函数重载?非要重载怎么做?
后端
码事漫谈1 小时前
浅谈C++与C语言二进制文件差异(从一次链接错误说起)
后端
空白诗3 小时前
mdcat 在 HarmonyOS 上的构建与适配
后端·安全·华为·rust·harmonyos
y***61313 小时前
SpringBoot集成Flowable
java·spring boot·后端
i***22074 小时前
springboot整合libreoffice(两种方式,使用本地和远程的libreoffice);docker中同时部署应用和libreoffice
spring boot·后端·docker
e***87704 小时前
windows配置永久路由
android·前端·后端
代码or搬砖4 小时前
SpringMVC的执行流程
java·spring boot·后端
极光代码工作室5 小时前
基于SpringBoot的流浪狗管理系统的设计与实现
java·spring boot·后端