二十一、Rust 反射 获取类型
不同于 java 中的反射,Rust 没有提供以往意义上的运行时反射,取而代之的是 “编译期反射”,如 类型分析、类型转换、类型签名。但即便如此,也已经能对 Rust元编程 提供很多助力了。
这种操作,主要通过 Any 来实现,Rust 中提供了 Any Trait,所有类型(含自定义类型)都自动实现了该特征;因此,通过它我们可以进行一些类似反射的功能;
实际上,在 Rust 早期版本中是提供了 Reflection 的,但是在 14年移除了相关代码,原因是:
- 反射打破了原有的封装原则,能任意访问结构体内容,不安全;
- 反射的存在使得代码过于臃肿,移除后编译器可以简化很多;
- 反射功能设计的比较弱,开发者对于是否在未来的版本中还拥有反射功能存疑;
另一篇 关于Rust为何不引入 Runtime Reflection ,大致信息如下:
- 不一定非要使用反射来实现, Rust中可以有更好的实现:
- 派生宏和Trait之间的配合,可以将实现从运行时转移到编译时;
例如,利用过程宏实现编译时反射,以实现依赖注入等功能 - https://github.com/dtolnay/reflect至于保留 Any 的原因:
- 在调试范型类型相关代码时,有TypeId会更方便,更容易给出正确的错误提示;
- 有利于编译器作出代码的优化;
Any源码简读
参看Any源码文档:
作为 &dyn Any (借用的 trait 对象),具有 is 和 downcast_ref 方法,可测试值是否为给定类型,并对类型的内部值进行引用;作为 &mut dyn Any,还有 downcast_mut 方法,用于获取内部值的 “可变引用” ;
Box<dyn Any> 具有 downcast 方法,该方法尝试转换为 Box<T>;也有称 “类型具象化”。但需注意,&dyn Any 仅限于测试值是否为具体的类型,而不能用于测试类型是否实现了 Trait;
总结就是 std::any 起到的作用有4个:
- 获得变量的类型
a.type_id(); - 判断变量是否是指定的具体类型
a.is::<String>()或TypeId::of::<String>() == a.type_id(); - 把any转换成指定类型
a.downcast_ref::<String>(); - 获取类型的名字
(_: &T) -> String { std::any::type_name::<T>().to_string() };
下面看一段 Any Trait 部分核心源码,可帮助更好理解 Any:
pub trait Any: 'static {fn type_id(&self) -> TypeId;
}// 获得变量的类型TypeId
// 为所有的T实现了Any
#[stable(feature = "rust1", since = "1.0.0")]
impl<T: 'static + ?Sized> Any for T {fn type_id(&self) -> TypeId { TypeId::of::<T>() }
}// 判断变量是否是指定类型
#[stable(feature = "rust1", since = "1.0.0")]
#[inline]
pub fn is<T: Any>(&self) -> bool {// Get `TypeId` of the type this function is instantiated with.let t = TypeId::of::<T>();// Get `TypeId` of the type in the trait object.let concrete = self.type_id();// Compare both `TypeId`s on equality.t == concrete
}// 把any转换成指定类型
#[stable(feature = "rust1", since = "1.0.0")]
#[inline]
pub fn downcast_ref<T: Any>(&self) -> Option<&T> {if self.is::<T>() {// SAFETY: just checked whether we are pointing to the correct typeunsafe {Some(&*(self as *const dyn Any as *const T))}} else {None}
}// 获取类型名字
pub const fn type_name<T: ?Sized>() -> &'static str {intrinsics::type_name::<T>()
}#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Debug, Hash)]
pub struct TypeId {t: u64,
}
补充说明:
-
Rust 中,所有拥有静态生命周期的类型都会实现Any,未来可能会考虑加入生命周期是非‘static的情况;
-
Rust 中,所有类型都有一个全局唯一的标识 TypeId(A
TypeIdrepresents a globally unique identifier for a type); -
这些 TypeId 都是通过调用 intrinsic 模块中定义的函数创建的,即 TypeId 的生成是由编译器的实现来决定的!
关于intrinsic 模块:
intrinsic 库函数是指:由编译器内置实现的函数,一般是具有如下特点的函数:
- 与CPU架构相关性很大,必须利用汇编实现或者利用汇编才能具备最高性能的函数;
- 和编译器密切相关的函数,由编译器来实现最为合适;
- 具体实现 - https://github.com/rust-lang/rust/blob/master/compiler/rustc_codegen_llvm/src/intrinsic.rs
Any基本使用
- examples/main.rs
use std::any::{Any, TypeId};struct Person {pub name: String,
}/// 判断是否为指定类型-1
fn is_string(s: &dyn Any) -> bool {TypeId::of::<String>() == s.type_id() // 获取TypeId
}/// 判断是否为指定类型-2
fn check_string(s: &dyn Any) {if s.is::<String>() {println!("It's a string!");} else {println!("Not a string...");}
}/// 转换Any为特定类型
fn print_if_string(s: &dyn Any) {if let Some(ss) = s.downcast_ref::<String>() {println!("It's a string({}): '{}'", ss.len(), ss);} else {println!("Not a string...");}
}/// 获取类型的名字
/// 但需注意, 此方式获取的名字不唯一
/// 如 type_name::<Option<String>> 可能返回 Option<String> 或 std::option::Option<std::string::String>
/// 同时, 编译器版本不同、可能返回值不同
fn get_type_name<T>(_: &T) -> String {std::any::type_name::<T>().to_string()
}fn main() {let p = Person { name: "John".to_string() };assert!(!is_string(&p));assert!(is_string(&p.name));check_string(&p);check_string(&p.name);print_if_string(&p);print_if_string(&p.name);println!("Type name of p: {}", get_type_name(&p));println!("Type name of p.name: {}", get_type_name(&p.name));
}
- 输出如下
Not a string...
It's a string!
Not a string...
It's a string(4): 'John'
Type name of p: 0_any::Person
Type name of p.name: alloc::string::String
Any适用场景
Rust 中的 Any 类似于 Java 中的 Object,可以传入任何拥有静态生命周期的类型;因此,当入参类型复杂,但后续又没有更多功能性操作时,就可以简化入参。例如,打印任何类型的值。
- examples/1_print_any.rs
use std::any::Any;
use std::fmt::Debug;#[derive(Debug)]
struct MyType {name: String,age: u32,
}fn print_any<T: Any + Debug>(value: &T) {let value_any = value as &dyn Any;if let Some(string) = value_any.downcast_ref::<String>() {println!("String ({}): {}", string.len(), string);} else if let Some(MyType { name, age }) = value_any.downcast_ref::<MyType>() {println!("MyType ({}, {})", name, age)} else {println!("{:?}", value)}
}fn main() {let ty = MyType {name: "Rust".to_string(),age: 30,};let name = String::from("Rust");print_any(&ty);print_any(&name);print_any(&30);
}
- 运行后输出
MyType (Rust, 30)
String (4): Rust
30
如上所示,不论 String、MyType 自定义类型、还是内置的 i32 类型,都可以被打印,只要他们实现了 Debug Trait;这也可以认为是Rust 中、一种函数重载的方式,在读取一些结构复杂的配置时,也可以直接使用 Any。
最后总结
Any Trait 并非常规意义上的 Reflection,而最多是编译期反射、且只启用了 “类型检查” 和 “类型转换”,并不检查结构的任意内容。
Any 符合零成本抽象,因为Rust只会针对调用该函数的相关类型生成代码,并且返回的是编译器内部的类型ID,没有额外开销;甚至可以直接使用 TypeId::of::<String>,从而没有了 dyn any 的动态绑定开销。
虽然 Rust 没有真正的 运行时 Reflection,但使用过程宏、仍可以实现大部分反射能够实现的功能,如上一节的 AOP 增强 !!
就这样,bye bye ~
参考资料
- https://www.jianshu.com/p/c4ef17bb1ca3
- https://rust.ffactory.org/std/any/index.html
- https://jasonkayzk.github.io/2022/11/24/Rust%E5%8F%8D%E5%B0%84%E4%B9%8BAny/
相关文章:
二十一、Rust 反射 获取类型
不同于 java 中的反射,Rust 没有提供以往意义上的运行时反射,取而代之的是 “编译期反射”,如 类型分析、类型转换、类型签名。但即便如此,也已经能对 Rust元编程 提供很多助力了。 这种操作,主要通过 Any 来实现&…...
Flutter Engine引擎概念
1.Flutter是Google提供的开源框架。 2.本身由C编写并兼容iOS(底层C)/Android(底层C)平台的FlutterEngine框架负责UI渲染、数据转移、调用DartVM虚拟机。 3.FlutterEngine框架由Skia图形库、Dart运行时、Flutter框架代码组成。Skia是用于图形绘制和文本显示的2D图形引擎库&#…...
【运行环境】加载资源的形式
相关资源:性能优化原则 1 加载资源的形式 html代码 媒体文件,如图片,视频等 javasccript css 2 加载资源的过程 DNS解析:域名-> ip地址 浏览器根据IP地址向服务器发送http 请求 服务器处理http 请求,并返回给浏览器…...
备战蓝桥杯Day40 - 第11届python组真题 - C跑步锻炼
一、题目描述 二、思路 1、使用datetime库中的方法可以很好的解决这个问题。 2、定义起始时间和结束时间,判断是否是周一或者是1号,结果res加上相应的里程数。 3、最后输出 res 即为本题答案。 三、代码实现 import datetimestart datetime.date(2…...
书生·浦语大模型第二期实战营第二课笔记和基础作业
来源: 作业要求:Homework - Demo 文档教程:轻松玩转书生浦语大模型趣味 Demo B站教程:轻松玩转书生浦语大模型趣味 Demo 1. 笔记 2.基础作业 2.1 作业要求 2.2 算力平台 2.3 新建demo目录,以及新建目录下的文件,下载模型参数 2.4 Intern…...
成功解决> 错误: 无效的源发行版:17
运行项目的时候出现下面的报错: Execution failed for task ‘:device_info_plus:compileDebugJavaWithJavac’. 错误: 无效的源发行版:17 原因:没有设置好自己项目的JDK版本 解决:1.检查自己项目的JDK版本 将自己的项目改为JDK 1…...
深度剖析:网络安全中的红蓝对抗策略
红蓝对抗 红蓝对抗服务方案 在蓝队服务中,作为攻击方将开展对目标资产的模拟入侵,寻找攻击路径,发现安全漏洞和隐患。除获取目标系统的关键信息(包括但不限于资产信息、重要业务数据、代码或管理员账号等)外&#x…...
Java异常处理之旅:解救迷失的程序员(二)
本系列文章简介: 在编程世界中,程序员们常常会遇到各种各样的问题和挑战。有时候,这些问题很容易解决,而有时候,它们却会让我们感到迷失和无助。 在这个旅程中,我们将探索Java异常处理的世界,解…...
网络安全介绍
网络安全是指网络系统的硬件、软件及其系统中的数据受到保护,不因偶然的或者恶意的原因而遭受到破坏、更改、泄露,系统能够连续可靠正常地运行,网络服务不中断。以下是一些网络安全相关的方面: 首先,随着科学技术的进…...
分享一个好看的APP下载分发页,App Store风格
分享一个好看的APP下载分发页,App Store风格 可以自动识别安卓和苹果哦! 内容直接可以页面上修改,所见即所得 下图是一个真实截图,想要的留下评论哦!...
C++ 获取数组大小、多维数组操作详解
获取数组的大小 要获取数组的大小,可以使用 sizeof() 运算符: 示例 int myNumbers[5] {10, 20, 30, 40, 50}; cout << sizeof(myNumbers);结果: 20为什么结果显示为 20 而不是 5,当数组包含 5 个元素时? 这…...
苹果电脑怎么彻底删除软件 苹果电脑卸载软件在哪里 cleanmymac x怎么卸载 mac废纸篓怎么删除
苹果电脑卸载软件的方法相对直观和简单,尤其是对于习惯使用Mac操作系统的用户来说。以苹果MacBook Pro为例,以下是卸载软件的详细步骤、使用方法、注意事项与建议。 一、卸载软件的详细步骤: 1. 打开Mac电脑,进入桌面,…...
STM32F407 FSMC并口读取AD7606
先贴一下最终效果图.这个是AD7606并口读取数据一个周期后的数据结果. 原始波形用示波器看是很平滑的. AD7606不知为何就会出现干扰, 我猜测可能是数字信号干扰导致的. 因为干扰的波形很有规律. 这种现象基本上可以排除是程序问题. 应该是干扰或者数字信号干扰,或者是数字和模拟…...
WebGPU vs. 像素流
在构建 Bzar 之前,我们讨论过我们的技术栈是基于在云上渲染内容的像素流,还是基于使用设备自身计算能力的本地渲染技术。 由于这种选择会极大地影响项目的成本、可扩展性和用户体验,因此在开始编写一行代码之前,从一开始就采取正确…...
Windows下docker-compose部署DolphinScheduler
参照:快速上手 - Docker部署(Docker) - 《Apache DolphinScheduler v3.1.0 使用手册》 - 书栈网 BookStack 下载源文件 地址:https://dolphinscheduler.apache.org/zh-cn/download/3.2.1 解压到指定目录,进入apache-dolphinscheduler-xxx-…...
微服务项目sc2024通用Base工程
1. cloud-provider-payment8001 2.pom文件 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"ht…...
git学习 1
打开自己想要存放git仓库的文件夹,右键打开git bush,用git init命令建立仓库 用 ls -a(表示全都要看,包括隐藏的)可以看到git仓库 也可以用 git clone 接github链接(点code选项里面会给链接,结尾是git的那个…...
HTML - 请你说一下如何阻止a标签跳转
难度级别:初级及以上 提问概率:55% a标签的默认语义化功能就是超链接,HTML给它的定位就是与外部页面进行交流,不过也可以通过锚点功能,定位到本页面的固定id区域去。但在开发场景中,又避免不了禁用a标签的需求,那么都有哪些方式可以禁用…...
【CV】ORB算法
1. ORB算法: 特点: 实现了旋转不变性、尺度不变性和计算效率高等特性。 旋转不变性: 通过计算关键点周围的梯度信息,确定关键点的主方向。将图像旋转到关键点的主方向,然后再提取BRIEF描述符,增强了旋转不…...
【算法】Cordic算法的原理及matlab/verilog应用
一、前言 单片机或者FPGA等计算能力弱的嵌入式设备进行加减运算还是容易实现,但是想要计算三角函数(sin、cos、tan),甚至双曲线、指数、对数这样复杂的函数,那就需要费些力了。通常这些函数的计算需要通者查找表或近似…...
(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)
题目:3442. 奇偶频次间的最大差值 I 思路 :哈希,时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况,哈希表这里用数组即可实现。 C版本: class Solution { public:int maxDifference(string s) {int a[26]…...
【Java学习笔记】Arrays类
Arrays 类 1. 导入包:import java.util.Arrays 2. 常用方法一览表 方法描述Arrays.toString()返回数组的字符串形式Arrays.sort()排序(自然排序和定制排序)Arrays.binarySearch()通过二分搜索法进行查找(前提:数组是…...
如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...
将对透视变换后的图像使用Otsu进行阈值化,来分离黑色和白色像素。这句话中的Otsu是什么意思?
Otsu 是一种自动阈值化方法,用于将图像分割为前景和背景。它通过最小化图像的类内方差或等价地最大化类间方差来选择最佳阈值。这种方法特别适用于图像的二值化处理,能够自动确定一个阈值,将图像中的像素分为黑色和白色两类。 Otsu 方法的原…...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
安卓基础(aar)
重新设置java21的环境,临时设置 $env:JAVA_HOME "D:\Android Studio\jbr" 查看当前环境变量 JAVA_HOME 的值 echo $env:JAVA_HOME 构建ARR文件 ./gradlew :private-lib:assembleRelease 目录是这样的: MyApp/ ├── app/ …...
云原生玩法三问:构建自定义开发环境
云原生玩法三问:构建自定义开发环境 引言 临时运维一个古董项目,无文档,无环境,无交接人,俗称三无。 运行设备的环境老,本地环境版本高,ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...
动态 Web 开发技术入门篇
一、HTTP 协议核心 1.1 HTTP 基础 协议全称 :HyperText Transfer Protocol(超文本传输协议) 默认端口 :HTTP 使用 80 端口,HTTPS 使用 443 端口。 请求方法 : GET :用于获取资源,…...
