宏 在 rust 中,我们一开始就在使用宏,例如 println!, vec!, assert_eq! 等。看起来宏和函数在使用时只是多了一个 !。 实际上这些宏都是声明式宏(也叫示例宏或macro_rules!),rust 还支持过程宏,过程宏为我们提供了强大的元编程工具。 声明式宏 声明式宏类似于 match 匹配。 这也是rust的宏是 Hygienic Macros 的体现。 宏 let mut v = my_vec!(); v.push(1); println!("{:?}", v); let v = my_vec![1, 2, 3, 4, 5]; println! v.push(3); v.push(4); v.push(5); v }; 它带上了我们在宏定义中的{},另外我们注意到println!
Rust之所以难,这也有很大的关系,把宏设计的比传统的宏要复杂一些,在记忆中宏就是代码替换,但Rust已经超出了这个定义。3年前,你在学习Rust,3年后,你还在学习Rust,可见Rust入坑要谨慎。 Rust宏的分类声明式宏和函数宏可以简单的理解为代码替换,也可以理解为传统的宏。声明式宏和过程宏的区别声明式宏按模式去匹配进行代码模板替换。按常规理解就行。 (config)数据库连接池UI 框架过程宏(Rust GUI)宏名所属框架用途示例说明#[component]yewYew 组件#[function_component] fn Comp()WebAssembly icedIced 组件#[derive(Widget)] struct BtnGUI 组件#[el]druidDruid 元素#[el] fn button() -> impl WidgetGUI 元素#[relm4: :widget]relm4Relm4 部件#[relm4::widget] impl WidgetGTK4 部件过程宏使用统计(2024年)根据 crates.io 下载量和 GitHub 星标,最流行的过程宏
过程宏是rust里的强大的武器,非常值得学习rust的人去掌握。但过程宏的编写有点难度,且文档也不太详细,最近也专门学习了下过程宏,算是有点收获,写下一点东西。 (1 + 1)); } 将得到正确的答案4。这是因为Rust的宏展开发生在语法分析阶段,此时编译器知道sqr! 对于宏编程,Rust中提供了几种过程宏的库操作支持,即: 1、Syn 它是基于TokenStream的一种语法分析过程,它并不很强大,需要自定义扩展一些宏,比如Rust中的函数和闭包等。 什么是过程宏? 过程宏(Procedure Macro)是Rust中的一种特殊形式的宏,它将提供比普通宏更强大的功能。方便起见,本文将Rust中由macro_rules!定义的宏称为规则宏以示区分。 结语 过程宏确实是rust里的黑魔法,希望这篇文章能帮助到一些人了解并使用过程宏,体会到rust的强大。 实例代码可以在这里看到。
概念 在编写过程宏时,经常需要对TokenStream 进行解析和处理。 而Syn库就是专门用于对TokenStream进行解析。 Syn is a parsing library for parsing a stream of Rust tokens into a syntax tree of Rust source code. Syn 用于将Rust tokens 解析为Rust 源码语法树。 实践 在过程宏使用Syn解析的流程: 定义自己存储结构 实现syn::parse::Parse 使用parse_macro_input!()生成源码树。 使用过程宏为struct添加hello方法: 使用parse_macro_input 解析到DeriveInput 获取struct name TokenStream 使用quote!
宏是Rust中的一种特殊函数,它可以接受代码片段作为输入,并根据需要生成代码片段作为输出。 本篇博客将深入探讨Rust中的声明宏,包括声明宏的定义、声明宏的特点、声明宏的使用方法,以及一些实际场景中的应用案例,以便读者全面了解Rust声明宏的魔力。 1. 宏来计算3和4的平方和,并将结果打印出来。 2.2 带模式匹配的声明宏例子 除了简单的替换,声明宏还可以使用模式匹配来更灵活地处理输入的代码片段。 结论 本篇博客深入探讨了Rust中的声明宏,包括声明宏的定义、声明宏的特点、声明宏的使用方法,以及一些实际场景中的应用案例。 声明宏是Rust中强大的元编程工具,通过模式匹配和代码生成,它使得代码更加灵活、易读和简洁。希望通过本篇博客的阐述,读者对Rust声明宏有了更深入的了解,并能在实际项目中灵活运用。谢谢阅读!
概念 宏的作用就是在编译期间对原代码进行扩展,实现目标功能。简单的说宏就是生成代码的代码。 . — The Rust Reference (你可以简单认为,过程宏是一个将原有AST语法树转换为另外一个AST语法树的函数) 个人理解,Rust 宏相比C++中的宏定义, 它提供了一种可用让开发人员更容易介入代码编译过程的入口 Rust 过程宏定义分三种 #[proc_macro] 函数似宏 和macro_rules! #[proc_macro_attribute] 属性宏 用于属性宏, 用在结构体、字段、函数等地方,为其指定属性等功能 其中 Derive Macro 派生宏和 Attribute Marco 宏的区别是 4. main.rs extern crate custom_derive; use custom_derive::log_attr; use custom_derive::make_hello; use
Rust那些事之过程宏 1.过程宏 过程宏是一种扩展Rust编译器和提供可用于扩展该语言的插件的方法。 过程宏的工作原理非常简单:取一段称为输入TokenStream的代码,将其转换为抽象语法树(ast),从输入处获得的内容构建一个新的TokenStream(使用syn::parse()方法),并将其作为输出代码注入编译器 2.derive宏 使用侧: #[derive(StructShow)] pub struct Point { x: f64, y: f64 } 我们需要使用proc-macro来实现该功能 将rust语法转位TokenStream返回给编译器即可。 ("{}", p); 今天写的这个例子比较简单,过程宏功能很强大,后续继续研究,期待一起探讨~ 本节完~
属性宏的基本概念 1.1 属性宏的定义 在Rust中,属性宏是一种特殊的宏,它允许开发者在代码上方添加自定义的属性,并在编译期间对代码进行处理。 1.2 属性宏的特点 属性宏在Rust中具有以下几个特点: 代码定制化处理:属性宏允许开发者在代码上方添加自定义的属性,并根据属性的输入对代码进行定制化处理。 代码安全性:属性宏生成的代码必须是合法的Rust代码,它们受到Rust编译器的类型检查和安全检查。这保证了宏生成的代码不会引入潜在的编译错误和安全漏洞。 2. 4. 属性宏的局限性 虽然属性宏在Rust中非常强大,但它也有一些局限性需要注意: 仅适用于特定项:属性宏只能应用于函数、结构体、枚举等特定的项,而不能应用于表达式等其他类型的代码。 希望通过本篇博客的阐述,读者对Rust属性宏有了更深入的了解,并能在实际项目中灵活运用。谢谢阅读!
本篇博客将深入探讨Rust中的宏,包括宏的定义、宏的分类、宏的使用方法,以及一些实际场景中的应用案例,以便读者全面了解Rust宏的神奇之处。 1. 过程宏:是一种更为高级的宏,它通过编写Rust代码来处理输入的代码,并在编译期间生成新的代码。 (1, 2, 3, 4, 5); println! 然后,在add函数上添加了#[check_arg]宏属性,这样宏就会对add函数的参数进行检查,确保它们大于10。 4. Rust宏的应用案例 Rust宏在实际开发中有许多应用案例,以下是一些常见的应用场景: 5.1 DRY原则(Don’t Repeat Yourself) 宏可以帮助我们遵循DRY原则,减少代码的重复编写
在本篇博客中,我们将深入探讨Rust中的派生宏,包括派生宏的定义、使用方法以及一些实际应用案例,以帮助读者充分了解派生宏的魅力。 1. 派生宏的基本概念 1.1 派生宏的定义 在Rust中,派生宏是一种特殊的宏,它允许开发者为自定义的数据类型自动实现trait。 代码安全性:派生宏生成的trait实现代码必须是合法的Rust代码,它们受到Rust编译器的类型检查和安全检查。这保证了派生宏生成的trait实现不会引入潜在的编译错误和安全漏洞。 2. 通过这个简单的派生宏,我们就能够轻松地为自定义的数据类型添加比较的功能,并使用派生的比较trait进行比较操作。 4. 派生宏的局限性 虽然派生宏在Rust中非常强大,但它也有一些局限性需要注意: trait的限制:派生宏只能自动实现由Rust标准库或第三方库定义的trait,无法自动实现用户自定义的trait。
场景开发过程宏时经常需要处理结构体或枚举体上的属性参数,如下 Command 结构体的 args 字段有属性 each = "arg",#[derive(Builder)]pub struct Command ], }, }, }, // 其它字段省略 宏入口
在本篇博客中,我们将深入探讨Rust中的类函数宏,包括类函数宏的定义、使用方法以及一些实际应用案例,以帮助读者充分了解类函数宏的魅力。 1. 类函数宏的基本概念 1.1 类函数宏的定义 在Rust中,类函数宏是一种特殊的宏,它允许开发者创建类似函数调用的宏,并在编译期间对代码进行生成和转换。 代码安全性:类函数宏生成的代码必须是合法的Rust代码,它们受到Rust编译器的类型检查和安全检查。这保证了宏生成的代码不会引入潜在的编译错误和安全漏洞。 2. 在宏的处理逻辑中,我们生成了一个代码块,其中包含了一些简单的变量声明和计算,并输出结果。 4. 结论 本篇博客中,我们深入探讨了Rust中的类函数宏,包括其定义、使用方法以及应用案例。
今天尝试下使用Rust中的过程宏来实现类似功能。
课程主题:《通过实战理解 Rust 宏》 课程时间: 2021年8月15日 20:30-21:30 课程介绍: 如果想用 Rust 开发大型目,或者学习大型项目代码,特别是框架级别的项目,那么 Rust 的宏机制肯定是一个必须掌握的技能。 语言强大的一个特点就是可以创建和利用宏,不过创建宏看起来挺复杂,常常令刚接触 Rust 的开发者生畏惧。 在本次公开课中帮助你理解 Rust Macro 的基本原理,学习如何创自已的 Rust 宏,以及查看源码学习宏的实现。 课程大纲 什么是 Rust 宏 什么是宏运行原理 如何创建 Rust 宏过程 阅读 datafuse 项目源码, 学习项目中宏的实现
Welcome to the #4 edition of Rust in Blockchain, the hypest newsletter about the hypest combination of featured Berlin Blockchain week, one of the biggest blockchain events of the year, and a number of Rust We held Rust in Blockchain events in Berlin as well as in Hangzhou, the videos for which (in Chinese) Rust in Blockchain has a new logo, and there are stickers! Parity Ethereum v2.5.9 was released v2.5.8 was released Solana PR: Clippy work towards Rust 1.38 by @
Rust中打印语句为什么使用宏?在Rust中,打印语句使用宏(例如println!和format!)的主要原因是为了在编译时进行字符串格式检查,并在不引入运行时开销的情况下提供更高的性能和安全性。 Rust宏允许在字符串中插入变量,而在编译时,编译器可以检查这些插值是否与实际的变量类型匹配。这有助于捕获潜在的格式化错误,防止运行时发生类型不匹配或其他问题。 在编译时,Rust会检查实际传递的参数是否与占位符的数量和类型匹配。2. 零成本抽象Rust中的宏提供了一种零成本的抽象。这意味着使用宏并不会引入运行时开销。在编译时,宏会被展开为实际的代码。 宏的简化定义。通过宏,可以将代码的抽象层次提高,同时不会影响性能。宏展开后,println!("Hello, world!") 和类似的宏使得代码更加灵活、可重用,并允许在编译时进行更多的优化。这是 Rust 中推崇的一种编程风格,有助于编写安全、高性能的代码。
5[1] 0.6> 3^5[1] 243> sqrt(9)[1] 3> abs(-3)[1] 3> log2(8)[1] 3> log10(100)[1] 2赋值<-> x <- 1+3> x[1] 4删除变量 > a <- -3> b <- 1> c <- 4> u <- 5+6> rm(b)> rm(u,c)> a[1] -3> bError: object 'b' not found> cfunction
Rust 宏定义,拯救运行时错误 如果你碰巧在 Rust 应用程序中使用 Lua ,你可以编写一个小宏,在 Lua 上强制执行它并运行 Lua 解释器以在编译时捕获相关错误。 作者以 Aerospike 分布式键值存储为,展开了如何通过 rust 宏中展开 lua 解析到 AST 上并校验 Aerospike 规则的方法。 C++ vs Rust :可变性和所有权 本文中,作者比较了 C++ 和 Rust 可变性,所有权等特性。 这篇文章可能对刚开始接触 Rust 的 C++ 开发人员特别有帮助。 侯盛鑫 坏姐姐 社区学习交流平台订阅: Rust.cc 论坛: 支持 rss 微信公众号:Rust 语言中文社区
[1, 2, 3, 4, 5];// 而不是:let mut v = Vec::new();v.push(1);v.push(2);v.push(3);v.push(4);v.push(5);4. Rust 宏的家族树scss 体验AI代码助手 代码解读复制代码Rust 宏├── 声明宏 (Declarative Macros)│ └── macro_rules! (sum 1, 2, 3, 4, 5)); // 15}1.4 实战案例:类型安全的单位转换rust 体验AI代码助手 代码解读复制代码macro_rules! [1, 2, 3, 4, 5];// ✅ 但要注意宏展开可能增加二进制大小// 如果宏在很多地方调用且生成大量代码,考虑改用函数常见误区误区 1:宏可以做任何事现实: 宏有明确的限制。 vs 函数:使用宏使用函数需要操作语法(如生成 trait 实现)简单的值计算需要在编译时验证某些条件运行时逻辑需要可变数量的参数类型固定的参数类型实现 DSL普通业务逻辑误区 4:不处理边界情况rust
改进 Rust 宏中的自动完成功能 自动完成是 IDE 提供的一种功能,可以帮助开发者在编写代码时快速找到正确的关键字和参数。在 Rust 宏中,自动完成功能可能会出现不准确或不完整的情况。 文章作者介绍了以下几种方法来改进 Rust 宏中的自动完成功能, 这样可以使你的宏在使用的过程中体验更好. 原文链接 https://blog.emi0x7d1.dev/improving-autocompletion-in-your-rust-macros/ IoT 和 Rust: 在 ESP 上连接 wifi 原文链接 https://dev.to/apollolabsbin/iot-with-rust-on-esp-connecting-wifi-4be6 Rust和 C++ 的两种动态分派方式 Rust Rust 和 C++ 动态分派的优缺点 Rust 优点: trait object 是类型安全的,这意味着编译器可以确保您不会调用不兼容的类型上的方法。 trait object 的性能开销很小。