【Rust 基础篇】Rust 属性宏:定制你的代码
导言
Rust是一门现代的、安全的系统级编程语言,它提供了丰富的元编程特性,其中属性宏(Attribute Macros)是其中之一。属性宏允许开发者在代码上方添加自定义的属性,并对代码进行定制化处理。在本篇博客中,我们将深入探讨Rust中的属性宏,包括属性宏的定义、使用方法以及一些实际应用案例,以帮助读者充分了解属性宏的魅力。
1. 属性宏的基本概念
1.1 属性宏的定义
在Rust中,属性宏是一种特殊的宏,它允许开发者在代码上方添加自定义的属性,并在编译期间对代码进行处理。属性宏使用proc_macro_attribute属性来定义,其基本形式如下:
extern crate proc_macro;use proc_macro::TokenStream;#[proc_macro_attribute]
pub fn attribute_macro(attr: TokenStream, item: TokenStream) -> TokenStream {// 宏的处理逻辑// ...
}
在上述例子中,我们使用proc_macro_attribute属性来定义了一个名为attribute_macro的属性宏。属性宏接受两个TokenStream参数:attr表示属性的输入,item表示应用该属性的代码块。在宏的处理逻辑中,我们可以根据attr和item对代码进行定制化处理,并返回一个TokenStream作为输出。
1.2 属性宏的特点
属性宏在Rust中具有以下几个特点:
-
代码定制化处理:属性宏允许开发者在代码上方添加自定义的属性,并根据属性的输入对代码进行定制化处理。这使得开发者可以根据需要修改代码的结构和行为。
-
编译期间执行:属性宏在编译期间执行,而不是运行时执行。这意味着宏生成的代码在编译时就已经确定,不会增加运行时的性能开销。
-
代码安全性:属性宏生成的代码必须是合法的Rust代码,它们受到Rust编译器的类型检查和安全检查。这保证了宏生成的代码不会引入潜在的编译错误和安全漏洞。
2. 属性宏的使用方法
2.1 简单的属性宏例子
让我们从一个简单的例子开始,创建一个属性宏用于在函数上方添加自定义的属性。
use proc_macro::TokenStream;#[proc_macro_attribute]
pub fn my_attribute(_attr: TokenStream, item: TokenStream) -> TokenStream {let mut result = item.to_string();result.push_str(" // This is my custom attribute!");result.parse().unwrap()
}#[my_attribute]
fn hello() {println!("Hello, attribute macro!");
}fn main() {hello();
}
在上述例子中,我们定义了一个名为my_attribute的属性宏。在宏的处理逻辑中,我们在函数上方添加了自定义的注释。在main函数中,我们应用了my_attribute宏到hello函数上。
2.2 带参数的属性宏例子
属性宏还可以带有参数,让我们创建一个带有参数的属性宏,用于生成不同类型的函数。
use proc_macro::TokenStream;#[proc_macro_attribute]
pub fn my_function(attr: TokenStream, item: TokenStream) -> TokenStream {let function_name = attr.to_string();let mut result = item.to_string();result.push_str(&format!("fn {}() {{", function_name));result.push_str("println!(\"This is a custom function generated by attribute macro!\"); }");result.parse().unwrap()
}#[my_function(hello)]
fn dummy() {}fn main() {hello();
}
在上述例子中,我们定义了一个名为my_function的属性宏,并使其带有一个参数attr,用于指定生成的函数名。在宏的处理逻辑中,我们根据参数生成了不同类型的函数。在main函数中,我们调用了通过my_function宏生成的hello函数。
3. 属性宏的应用案例
3.1 自定义数据结构
属性宏可以用于定制化地生成自定义数据结构。让我们通过一个例子来演示如何使用属性宏生成一个自定义的数据结构。
use proc_macro::TokenStream;#[proc_macro_attribute]
pub fn my_struct(attr: TokenStream, item: TokenStream) -> TokenStream {let struct_name = attr.to_string();let mut result = item.to_string();result.push_str(&format!("struct {} {{", struct_name));result.push_str("data: i32 }");result.parse().unwrap()
}#[my_struct(Point)]
fn dummy() {}fn main() {let point = Point { data: 10 };println!("Data: {}", point.data); // 输出:Data: 10
}
在上述例子中,我们定义了一个名为my_struct的属性宏,并使其带有一个参数attr,用于指定生成的数据结构名。在宏的处理逻辑中,我们根据参数生成了一个自定义的数据结构。在main函数中,我们通过my_struct宏生成了Point结构体,并创建了一个Point的实例,并输出其中的字段。
3.2 条件编译
属性宏可以用于实现条件编译,让我们通过一个例子来演示如何使用属性宏实现条件编译。
use proc_macro::TokenStream;#[proc_macro_attribute]
pub fn my_feature(_attr: TokenStream, item: TokenStream) -> TokenStream {let mut result = item.to_string();#[cfg(feature = "my_feature")]result.push_str("fn my_function() { println!(\"my_feature is enabled!\"); }");result.parse().unwrap()
}#[my_feature]
fn main() {#[cfg(feature = "my_feature")]my_function();
}#[cfg(not(feature = "my_feature"))]
fn my_function() {println!("my_feature is not enabled!");
}
在上述例子中,我们定义了一个名为my_feature的属性宏,用于在代码中添加条件编译的逻辑。在宏的处理逻辑中,我们根据cfg属性来判断是否启用了特定的feature,并根据不同情况生成了不同的代码。在main函数中,我们通过my_feature宏来控制是否调用my_function函数。
4. 属性宏的局限性
虽然属性宏在Rust中非常强大,但它也有一些局限性需要注意:
-
仅适用于特定项:属性宏只能应用于函数、结构体、枚举等特定的项,而不能应用于表达式等其他类型的代码。
-
无法修改输入项:属性宏只能生成新的代码,而不能修改输入项的内容。例如,无法在函数内部添加新的语句或修改函数的签名。
-
不支持模式匹配:与声明宏不同,属性宏不能进行模式匹配,只能对整个输入项进行处理。
结论
本篇博客深入探讨了Rust中的属性宏,包括属性宏的定义、使用方法以及一些实际应用案例。属性宏允许开发者在代码上方添加自定义的属性,并在编译期间对代码进行处理,从而实现代码的定制化。属性宏在Rust中是非常强大且有用的元编程工具,它为开发者提供了更多的灵活性和可定制性。希望通过本篇博客的阐述,读者对Rust属性宏有了更深入的了解,并能在实际项目中灵活运用。谢谢阅读!
相关文章:
【Rust 基础篇】Rust 属性宏:定制你的代码
导言 Rust是一门现代的、安全的系统级编程语言,它提供了丰富的元编程特性,其中属性宏(Attribute Macros)是其中之一。属性宏允许开发者在代码上方添加自定义的属性,并对代码进行定制化处理。在本篇博客中,…...
2023-08-04力扣今日三题
链接: 剑指 Offer 35. 复杂链表的复制 题意: 如题 解: 看题研究了好一阵,指针map 实际代码: #include<bits/stdc.h> using namespace std; class Node { public:int val;Node* next;Node* random;Node(in…...
从HTTP代理到Socks5代理:网络安全与爬虫的进化之路
一、HTTP代理:简介与特点 HTTP代理是一种最早的代理技术,通过HTTP协议转发网络请求。它能够隐藏用户的真实IP地址,实现匿名访问,为爬虫应用提供了最基本的代理功能。 HTTP代理只支持TCP协议,对于实时数据传输和UDP协议…...
数学建模-元胞自动机
clc clear n 300; % 定义表示森林的矩阵大小 Plight 5e-6; Pgrowth 1e-2; % 定义闪电和生长的概率 UL [n,1:n-1]; DR [2:n,1]; % 定义上左,下右邻居 vegzeros(n,n); % 初始化表示森林的矩阵 imh ima…...
化学合成有机化学 | 逆合成分析软件/数据库汇总
化合物逆合成路线设计软件是一类用于辅助化学家设计化合物合成路线的工具。这些软件通常基于化学知识和反应数据库,能够根据目标化合物的结构和性质,提供合成路线的建议和优化方案。以下是一些常见的化合物逆合成路线设计软件: IntSynth&…...
无涯教程-jQuery - Selectable选择函数
选择能力功能可与JqueryUI中的交互一起使用。此功能可在任何DOM元素上启用选择能力功能。用光标绘制一个框以选择项目。按住Ctrl键可进行多个不相邻的选择。 Select able - 语法 $( "#selectable" ).selectable(); Select able - 示例 以下是一个简单的示例&…...
MySQL修改root密码
1、使用set password命令 mysql -uroot mysql> use mysql mysql> set password for rootlocalhost PASSWORD(newpass); mysql> flush privileges; mysql> select user,host,password from user; mysql> exit 2、使用update user表 mysql -uroot mysql> …...
vue获取近七天、月份、年份的起始日和结束日
vue获取近七天的起始日和结束日 例如:startDate: 2023-07-29 endDate: 2023-08-04 data() {return {startDate: null,endDate: null} }, mounted() {this.calculateDateRange(); }, methods: {calculateDateRange() {var currentDate new Date();var startDate …...
android AIDL 学习使用
在android studio 2023.2中使用 1、在buidl.gradle增加以下配置,然后同步。不增加这些配置,创建aidl时显示为灰色,不能创建 buildFeatures {compose true// Disable unused AGP featuresbuildConfig falseaidl truerenderScript falseresVal…...
学习笔记|C251|STC32G单片机视频开发教程(冲哥)|第三集:开发环境搭建和程序下载
文章目录 1.STC-ISP软件的下载2.STC32手册下载3.PDF阅读器下载4.学会PDF阅读器查阅手册5.跟着手册搭建C251开发环境Tips:如何同时安装Keil的C51、C251和MDK 6.程序包的下载7.第一个工程的编译和下载 原作者/主讲人:冲哥 原始视频地址 1.STC-ISP软件的下载 STC-ISP …...
【数据可视化】(二)数据探索组件
目录 0.简介 一、数据模式与数据组织 1、数据的定义 2、数据库的定义 3、什么是数据模式? 4、数据模式举例 5、什么是数据纲要? 6、数据组织的层次 二、矢量数据 1、什么是矢量数据?...
Go to Play Maimai DX 2023牛客暑期多校训练营5 G
登录—专业IT笔试面试备考平台_牛客网 题目大意:给出一长度为n的仅由1,2,3,4组成的数组和一整数k,求一个最短的区间使得1,2,3,4至少各有一个,且4的数量>k 1<k<n<1e5 思路:用双指针l,r维护合法区间&…...
HTML基础铺垫
😊HTML基础铺垫 👻前言📜HTML文档结构🎭头部head🥏标题title标记🥏元信息meta标记 🎭主体body🥏body标记🥏body标记属性 🎭HTML基本语法🥏标记类型…...
【Vue3项目实战】vue3项目基于el-menu封装左侧菜单栏组件
文章目录 概述一、先看效果1.1 静态效果1.2 动态效果 二、核心思路三、全量代码3.1 文件目录结构3.2 /sidebar/index.vue 中3.3 /sidebar/sidebarItem.vue 中3.4 路由表结构 四、代码讲解五、SVG组件六、系列文章友链1、[配置husky、stylelint、commitlint,实现git提…...
MySQL正则表达式检索数据
目录 一、使用正则表达式进行基本字符匹配 1.使用regexp关键字 2.使用正则表达式 . 二、进行OR匹配 1.为搜索两个串之一,使用 | 2.匹配几个字符之一[] 3.匹配范围 4.匹配特殊字符 过滤数据允许使用匹配、比较、通配符操作来寻找数据,但是随…...
vite+ts+vue3 prettier.config.js 不生效问题解决
vitetsvue3 prettier.config.js 不生效问题解决 我在做项目的时候 我发现 我的vscode prettier插件 坏了 我自动格式化代码也开了 就是不给我格式化, 我已经写了prettier.config.js这个配置 也 npm i prettier 下载了就是不生效 后来我发现是因为 这个package.json 里的 “ty…...
Java源码规则引擎:jvs-rules 8月新增功能介绍
JVS-rules是JAVA语言下开发的规则引擎,是jvs企业级数字化解决方案中的重要配置化工具,核心解决业务判断的配置化,常见的使用场景:金融信贷风控判断、商品优惠折扣计算、对员工考核评分等各种变化的规则判断情景。 8月是收获的季节…...
2023年第三届工业自动化、机器人与控制工程国际会议 | IET独立出版 | EI检索
会议简介 Brief Introduction 2023年第三届工业自动化、机器人与控制工程国际会议(IARCE 2023) 会议时间:2023年10月27 -30日 召开地点:中国成都 大会官网:www.iarce.org 2023年第三届工业自动化、机器人与控制工程国际…...
14.2.2 【Linux】software, hardware RAID
磁盘阵列分为硬件与软件。所谓的硬件磁盘阵列是通过磁盘阵列卡来达成阵列的目的。磁盘阵列卡上面有一块专门的芯片在处理 RAID 的任务,因此在性能方面会比较好。在很多任务 (例如 RAID 5 的同位检查码计算) 磁盘阵列并不会重复消耗原本系统的…...
(学习笔记-进程管理)进程
进程 我们编写的代码只是一个存储在硬盘的静态文件,通过编译后会生成二进制可执行文件,当我们运行这个可执行文件后,它会被装载到内存中,接着CPU会执行程序中的每一条指令,那么这个运行中的程序就被称为进程。 现在我…...
椭圆曲线密码学(ECC)
一、ECC算法概述 椭圆曲线密码学(Elliptic Curve Cryptography)是基于椭圆曲线数学理论的公钥密码系统,由Neal Koblitz和Victor Miller在1985年独立提出。相比RSA,ECC在相同安全强度下密钥更短(256位ECC ≈ 3072位RSA…...
Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...
ABAP设计模式之---“简单设计原则(Simple Design)”
“Simple Design”(简单设计)是软件开发中的一个重要理念,倡导以最简单的方式实现软件功能,以确保代码清晰易懂、易维护,并在项目需求变化时能够快速适应。 其核心目标是避免复杂和过度设计,遵循“让事情保…...
#Uniapp篇:chrome调试unapp适配
chrome调试设备----使用Android模拟机开发调试移动端页面 Chrome://inspect/#devices MuMu模拟器Edge浏览器:Android原生APP嵌入的H5页面元素定位 chrome://inspect/#devices uniapp单位适配 根路径下 postcss.config.js 需要装这些插件 “postcss”: “^8.5.…...
基于 TAPD 进行项目管理
起因 自己写了个小工具,仓库用的Github。之前在用markdown进行需求管理,现在随着功能的增加,感觉有点难以管理了,所以用TAPD这个工具进行需求、Bug管理。 操作流程 注册 TAPD,需要提供一个企业名新建一个项目&#…...
STM32HAL库USART源代码解析及应用
STM32HAL库USART源代码解析 前言STM32CubeIDE配置串口USART和UART的选择使用模式参数设置GPIO配置DMA配置中断配置硬件流控制使能生成代码解析和使用方法串口初始化__UART_HandleTypeDef结构体浅析HAL库代码实际使用方法使用轮询方式发送使用轮询方式接收使用中断方式发送使用中…...
省略号和可变参数模板
本文主要介绍如何展开可变参数的参数包 1.C语言的va_list展开可变参数 #include <iostream> #include <cstdarg>void printNumbers(int count, ...) {// 声明va_list类型的变量va_list args;// 使用va_start将可变参数写入变量argsva_start(args, count);for (in…...
高考志愿填报管理系统---开发介绍
高考志愿填报管理系统是一款专为教育机构、学校和教师设计的学生信息管理和志愿填报辅助平台。系统基于Django框架开发,采用现代化的Web技术,为教育工作者提供高效、安全、便捷的学生管理解决方案。 ## 📋 系统概述 ### 🎯 系统定…...
JS红宝书笔记 - 3.3 变量
要定义变量,可以使用var操作符,后跟变量名 ES实现变量初始化,因此可以同时定义变量并设置它的值 使用var操作符定义的变量会成为包含它的函数的局部变量。 在函数内定义变量时省略var操作符,可以创建一个全局变量 如果需要定义…...
高效的后台管理系统——可进行二次开发
随着互联网技术的迅猛发展,企业的数字化管理变得愈加重要。后台管理系统作为数据存储与业务管理的核心,成为了现代企业不可或缺的一部分。今天我们要介绍的是一款名为 若依后台管理框架 的系统,它不仅支持跨平台应用,还能提供丰富…...
