news 2026/4/17 12:12:20

开发常用 宏

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
开发常用 宏

1、Rust 标准库 derive 宏与第三方 derive 宏的核心区别

二者本质都是编译期自动生成代码的声明宏,但在依赖来源、功能定位、实现方式、稳定性等核心维度有本质差异,直接决定了使用方式、适用场景和工程依赖成本。

一、核心维度对比表

对比维度标准库derive宏第三方derive宏
依赖来源随Rust编译器和标准库内置,无需额外引入依赖来自crates.io社区库,必须在Cargo.toml添加依赖并启用对应特性
功能定位提供语言级、通用基础能力,覆盖数据类型核心操作聚焦业务与开发效率,解决标准库未覆盖的特定场景需求
实现机制多为编译器原生支持,部分由标准库内部过程宏实现几乎均基于过程宏(proc-macro),编译期动态生成代码
稳定性与兼容性随Rust语言版本严格保证向后兼容,稳定性极高由社区维护,需关注版本更新、破坏性变更,兼容性依赖库的维护规范
维护主体Rust官方团队独立开发者、社区组织或公司
使用成本零额外成本,直接使用需管理依赖版本、编译时长略有增加,部分复杂宏需学习额外配置
代表示例DebugCloneCopyPartialEqEqPartialOrdOrdHashDefaultSerialize/Deserialize(serde)、Builder(derive_builder)、Getters(derive_getters)

二、关键差异详细解析

1. 依赖与使用门槛的差异

标准库宏:开箱即用

直接通过#[derive()]属性使用,无需修改Cargo.toml,是Rust项目的基础能力。

// 无需任何额外依赖,直接派生标准库trait#[derive(Debug, Clone, PartialEq)]structPoint{x:i32,y:i32,}
第三方宏:依赖先行

必须先在Cargo.toml声明依赖,部分还需开启derive等特性,才能使用对应的宏。

# 例:使用serde的序列化宏,必须添加依赖并启用derive特性 [dependencies] serde = { version = "1.0", features = ["derive"] }
// 依赖引入后,才能使用第三方derive宏useserde::{Serialize,Deserialize};#[derive(Debug, Serialize, Deserialize)]structUser{id:u64,name:String,}

2. 功能定位与覆盖范围的差异

  • 标准库宏:聚焦语言核心语义,解决所有Rust项目都可能用到的通用问题,比如打印调试、值拷贝、相等比较、哈希计算、默认值初始化,是构建所有数据类型的基石,不涉及业务逻辑。
  • 第三方宏:聚焦效率提升与场景化解决方案,是标准库功能的延伸,比如自动生成构建者模式、序列化反序列化、自动实现Getter/Setter、数据校验、ORM映射等,针对特定开发痛点,大幅减少重复代码。

3. 实现机制的本质差异

标准库宏

一部分是编译器硬编码支持,编译器直接识别这些trait并生成对应实现代码,编译效率极高;另一部分由标准库内部的过程宏实现,但对用户完全透明,无需感知底层逻辑。
其生成的代码严格遵循Rust语言规范,和手动实现的trait代码逻辑完全一致,无额外副作用。

第三方宏

大部分基于过程宏(Procedural Macros)实现,过程宏是Rust的一类特殊宏,能在编译期读取被标注的结构体/枚举的语法树(AST),动态生成任意合法的Rust代码。
这种机制让第三方宏功能极具扩展性,但也意味着:

  • 编译时会额外执行宏的代码生成逻辑,小幅增加编译时长
  • 宏的代码质量、安全性完全由维护者保证,可能存在潜在bug。

4. 稳定性与工程风险的差异

标准库宏
  • 官方维护,遵循Rust的语义化版本和向后兼容承诺,几乎不会出现破坏性变更;
  • 无依赖冲突风险,是项目最稳定的基础组件,适用于所有对稳定性要求高的场景。
第三方宏
  • 社区维护,更新节奏不固定,大版本升级可能存在破坏性变更,需要手动适配;
  • 存在依赖冲突、版本兼容问题(比如多个库依赖同一个基础库的不同版本);
  • 小众宏可能存在维护停滞、安全漏洞的风险,生产环境需谨慎选择。

2、标准库

1. PartialEq & Eq(相等性判断)

  • 作用:
    PartialEq实现== / !=运算符,Eq是其强化版(表示 “完全等价”,无 NaN 这类特殊值)。
  • 适用条件:
    结构体 / 枚举的所有字段都实现了PartialEq(Eq要求所有字段实现Eq)。
#[derive(Debug, PartialEq, Eq)]structAbc{a:i32,name:String}fnmain(){leta=Abc{a:1,name:"abc".to_string()};letb=Abc{a:1,name:String::from("abc")};println!("{:?}",a==b);// true}

2. Default

  • 作用:
    实现Default trait,通过T::default()生成默认值,常用于配置、初始化。
  • 适用条件:
    所有字段实现Default(或手动指定默认变体 / 值)。
#[derive(Debug, Default)]structConfig{timeout:u64,// u64默认0max_retries:u8,// u8默认0name:String,enable_ssl:bool,// bool默认false}fnmain(){leta=Config{timeout:5,enable_ssl:true,..Default::default()};println!("{:?}",a);// Config { timeout: 5, max_retries: 0, name: "", enable_ssl: true }}

3. Hash(哈希计算)

  • 作用:
    实现Hash trait,生成哈希值。 只有能 生成Hash值 才能作为 HashMap / HashSet 的 key
  • 注意:
    需搭配PartialEq
usestd::collections::HashMap;usestd::hash::Hash;#[derive(Debug, PartialEq, Eq, Hash)]structUserId(u64);// 单字段结构体fnmain(){letmutuser_map=HashMap::new();user_map.insert(UserId(1001),"Alice");}

3、第三方

1. Serialize & Deserialize(serde,序列化 / 反序列化)

  • 作用:
    Rust 序列化事实标准,支持 JSON、Bincode、YAML 等格式。

  • 依赖:
    [dependencies]
    serde = { version = “1.0”, features = [“derive”] }
    serde_json = “1.0” # 用于处理 JSON 格式

基本用法

useserde::{Deserialize,Serialize};#[derive(Serialize, Deserialize, Debug)]structUser{name:String,age:u32,email:String,aaaBBB:i8,bbb_ccc:i8,}fnmain(){letuser=User{name:"Alice".to_string(),age:30,email:"alice@example.com".to_string(),aaaBBB:12,bbb_ccc:5,};// 序列化为 JSON 字符串letjson=serde_json::to_string(&user).unwrap();println!("序列化结果: {}",json);// 从 JSON 字符串反序列化letuser_from_json:User=serde_json::from_str(&json).unwrap();println!("反序列化结果: {:?}",user_from_json);// 序列化结果: {"name":"Alice","age":30,"email":"alice@example.com","aaaBBB":12,"bbb_ccc":5}// 反序列化结果: User { name: "Alice", age: 30, email: "alice@example.com", aaaBBB: 12, bbb_ccc: 5 }}

隐藏字段

#[derive(Serialize, Deserialize, Debug)]structConfig{username:String,#[serde(skip_serializing)]// 序列化时忽略此字段password:String,#[serde(skip_deserializing)]// 反序列化时忽略此字段temp_dir:String,}

反序列化 零值处理

让缺失的字段在反序列化时自动赋值为类型的“零值”(比如 0 for i32, “” for String),你需要使用 Serde 的 default 属性。

useserde::{Deserialize,Serialize};#[derive(Serialize, Deserialize, Debug)]structUser{name:String,#[serde(skip_serializing, default)]// 序列化时忽略此字段age:u32,}fnmain(){letuser=User{name:"Alice".to_string(),age:20,};// 序列化为 JSON 字符串letjson=serde_json::to_string(&user).unwrap();println!("序列化结果: {}",json);// 序列化结果: {"name":"Alice"}// 从 JSON 字符串反序列化letuser_from_json:User=serde_json::from_str(&json).unwrap();println!("反序列化结果: {:?}",user_from_json);// 反序列化结果: User { name: "Alice", age: 0 }}

为字段起别名

  • 单个字段重命名:使用rename指定新的名称。
  • 批量重命名:使用rename_all配合命名规范(如 小驼峰snake_case, 大驼峰camelCase)一次性修改所有字段。
useserde::{Deserialize,Serialize};#[derive(Serialize, Deserialize, Debug)]#[serde(rename_all ="camelCase")]// 将所有字段名转换为驼峰命名法structUser{#[serde(rename ="nickname")]// 单独为某个字段指定别名name:String,aaaBBB:i8,bbb_ccc:i8,}fnmain(){letuser=User{name:"Alice".to_string(),aaaBBB:12,bbb_ccc:5,};// 序列化为 JSON 字符串letjson=serde_json::to_string(&user).unwrap();println!("序列化结果: {}",json);// 序列化结果: {"nickname":"Alice","aaaBBB":12,"bbbCcc":5}// 从 JSON 字符串反序列化letuser_from_json:User=serde_json::from_str(&json).unwrap();println!("反序列化结果: {:?}",user_from_json);// 反序列化结果: User { name: "Alice", aaaBBB: 12, bbb_ccc: 5 }}

空值处理

序列化时跳过空值:使用 skip_serializing_if。常用于 Option,当值为 None 时,该字段不会出现在输出中。

#[derive(Serialize, Deserialize, Debug, Default)]structProfile{name:String,#[serde(skip_serializing_if ="Option::is_none")]// 如果 avatar 是 None,序列化时跳过该字段avatar:Option<String>,}
  • 条件性跳过序列化
    除了跳过空值,你还可以根据自定义的条件函数来决定是否跳过某个字段。skip_serializing_if 接受一个返回 bool 的函数。
#[derive(Serialize, Deserialize, Debug)]structData{value:i32,#[serde(skip_serializing_if ="should_skip_message")]message:String,}// 自定义条件函数fnshould_skip_message(msg:&String)->bool{msg.is_empty()||msg.starts_with("internal")}

常用 Serde 属性速查表

属性作用示例
#[serde(rename = "new_name")]为字段或枚举变体指定别名#[serde(rename = "id")]
#[serde(rename_all = "camelCase")]批量重命名所有字段snake_case,PascalCase
#[serde(skip_serializing)]序列化时跳过该字段敏感信息如密码
#[serde(skip_deserializing)]反序列化时跳过该字段运行时生成的临时数据
#[serde(skip_serializing_if = "path")]满足条件时跳过序列化skip_serializing_if = "Option::is_none"
#[serde(default)]反序列化时使用默认值字段缺失时设为false0
#[serde(deny_unknown_fields)]禁止反序列化未知字段,遇到则报错常用于严格配置文件解析
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/12 2:59:35

redis源码deps目录

文章目录 fast-floatfpconvhdr_histogramhiredisjemalloclinenoiselua REDIS源码里deps目录是做什么的&#xff1f;从阅读deps目录&#xff0c;我们可以感觉到redis&#xff0c;真的是对性能的追求达到了极致&#xff0c;尽一切可能去优化性能。 deps主要存放的是依赖的第三方库…

作者头像 李华
网站建设 2026/4/16 11:53:06

小程序毕设选题推荐:微信小程序的个人运动健康管理平台基于SpringBoot+Vue的个人运动健康小程序的设计与实现【附源码、mysql、文档、调试+代码讲解+全bao等】

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/4/15 18:36:11

【Vue知识点总结】API封装全指南:参数类型、场景选择与企业实战

一、API 封装的核心价值 在企业级 Vue 项目中,API 封装不仅是为了代码整洁,更是为了: 统一接口管理:集中维护所有后端接口地址 参数标准化:规范请求/响应数据格式 错误处理:统一拦截异常和状态码 提高复用性:减少重复代码 便于测试:模块化接口更易mock和测试 二、API …

作者头像 李华
网站建设 2026/4/16 22:42:40

全球股市估值与基因治疗的医疗保险政策框架

全球股市估值与基因治疗的医疗保险政策框架 关键词:全球股市估值、基因治疗、医疗保险政策、金融与医疗融合、投资分析 摘要:本文深入探讨了全球股市估值与基因治疗的医疗保险政策框架之间的关系。首先介绍了研究的背景、目的、预期读者等内容。接着阐述了全球股市估值和基因…

作者头像 李华
网站建设 2026/4/16 21:25:11

Claude:开启AI原生应用领域新时代

Claude与AI原生应用&#xff1a;从工具革命到生态重构 关键词 Claude、AI原生应用、Constitutional AI、长上下文处理、生成式AI、prompt工程、安全可控性 摘要 当我们谈论"AI原生应用"时&#xff0c;我们究竟在谈论什么&#xff1f;是将ChatGPT嵌入App的"A…

作者头像 李华
网站建设 2026/4/10 21:29:35

亲测高中自习室全托供应商,案例复盘分享

亲测高中自习室全托供应商&#xff0c;案例复盘分享在高中教育焦虑持续升温的背景下&#xff0c;越来越多家长开始重视“自习室”这一新兴学习场景。不同于传统补习班的“讲授式”学习&#xff0c;现代高中自习室强调自主性、系统性和精准性。近年来&#xff0c;【高中自习室】…

作者头像 李华