Solana 学习开发之旅

2025年04月03日更新 33 人订阅
原价: ¥ 20 限时优惠
专栏简介 【Solana】使用 CLI 创建 SPL 标准的 Token 以及基础使用 【Solana】完善 SPL Token 名称和 Logo 【Solana】创建 SPL 标准的 NFT 以及完善 metadata 【Solana】一些基本的js脚本 【Solana】Anchor 框架使用笔记 【Solana】Anchor 示例:通过 CPI 实现 Sol 转账与手续费收取 Solana Hello World: 安装与开发指南 Solana 与 Rust 算术入门—从 Solidity 到 Anchor Solana Anchor 程序接口定义语言(IDL) Solana Anchor 框架下的 Require 与自定义错误 Solana 程序:支持升级与无构造函数实现 Solidity 开发者必知的 Rust 语法基础 Rust 的独特语法解析 Rust 类函数宏解析 Rust 结构体、属性宏与自定义派生宏 Rust 与 Solana 中的可见性及模块化复用 Solana 中的时钟与其他区块变量 Solana 系统变量详解 Solana 日志、事件日志与历史交易查询 Solana 中的Tx.origin、msg.sender 和 onlyOwner Solana 计算单元与交易费用概述 Solana 与 Anchor 中的账户初始化 Solana 计数器教程:账户数据的读写 使用 Solana Web3.js 和 Anchor 读取账户数据 在 Solana 中实现映射表与嵌套映射表 Solana 存储成本、最大容量与账户调整 在 Anchor 中读取账户余额:Solana 的 address(account).balance Solana 中的函数修饰符与 Fallback 函数:为何不存在 Solana 中的 SOL 转移与分割:取代 msg.value 的设计 使用不同签名者修改账户:Solana 中的权限控制 PDA 与密钥对账户:Solana 中的地址与权限模型 Anchor 中的 init_if_needed 与重新初始化攻击防范 Solana 中的 Multicall:批处理交易和交易大小的限制 Solana 中的 Owner 和 Authority 删除和关闭 Solana 中的账户和程序 在 Anchor 中的 #[derive(Accounts)] 不同类型的账户 在链上读取另一个 Anchor 程序的账户数据 Anchor 中的跨程序调用

Rust 类函数宏解析

  • 0xE
  • 发布于 2025-03-24 13:17
  • 阅读 927

本文将阐释 Rust 中函数与类函数宏的区别,例如为何 msg! 后带有感叹号 !。我们将深入探讨这种语法的意义及其应用。

本文将阐释 Rust 中函数与类函数宏的区别,例如为何 msg! 后带有感叹号 !。我们将深入探讨这种语法的意义及其应用。

Rust 作为强类型语言,不支持函数接受任意数量的参数。例如,Python 的 print 函数可以灵活处理:

print(1) # 单个参数
print(1, 2) # 两个参数
print(1, 2, 3) # 三个参数

在 Rust 中,带有 ! 的语法(如 println! 或 msg!)表示这是一个类函数宏,而非普通函数。


函数与宏的区别

Rust 中用于输出的普通函数是 std::io::stdout().write,它只接受单一字节切片参数:

use std::io::Write;

fn main() {
    std::io::stdout().write(b"Hello, world!\n").unwrap(); // 输出字节切片
}

注意,write 是函数,没有 !。若尝试像 Python 那样传入多个参数,会编译失败:

use std::io::Write;

fn main() {
    std::io::stdout().write(b"1\n").unwrap();
    // std::io::stdout().write(b"1", b"2\n").unwrap();   // 编译失败:参数数量不匹配
}

若想支持任意参数数量,一个低效的办法是为每种参数个数编写特定函数:

use std::io::Write;

fn print1(arg1: &[u8]) { // 输出单个参数
    std::io::stdout().write(arg1).unwrap();
}

fn print2(arg1: &[u8], arg2: &[u8]) { // 输出两个参数
    let combined = [arg1, b" ", arg2].concat();
    std::io::stdout().write(&combined).unwrap();
}

fn print3(arg1: &[u8], arg2: &[u8], arg3: &[u8]) { // 输出三个参数
    let combined = [arg1, b" ", arg2, b" ", arg3].concat();
    std::io::stdout().write(&combined).unwrap();
}

fn main() {
    print1(b"1\n");
    print2(b"1", b"2\n");
    print3(b"1", b"2", b"3\n");
}

这种方法显然繁琐且不可扩展。Rust 的类函数宏通过动态生成代码解决了这一问题。

宏的本质

Rust 宏以 Rust 代码为输入,在编译时将其扩展为更复杂的代码。例如,println! 可以根据参数数量自动生成相应的输出逻辑,避免手动编写多个函数。

可用 cargo-expand 查看宏的扩展结果,但其输出冗长,此处不展示。


将宏视为黑盒

对于开发者而言,通常无需深入理解宏的内部实现。由库提供的宏(如 println!、msg!)使用方便,但手动编写宏较为复杂,涉及 Rust 代码的解析。


Rust 中的宏类型

Rust 支持多种宏类型,以下是我们关注的几种:

  1. 类函数宏(Function-like Macro) 示例:println!、msg!、declare_id!通过 ! 调用,类似函数但更灵活。
  2. 属性宏(Attribute-like Macro) 示例:#[program]以属性形式附加到代码项上,修改其行为。
  3. 自定义派生宏(Custom Derive Macro) 示例:#[derive(Accounts)]为结构体或枚举自动实现特定功能。

以下是 Anchor 生成的程序示例:

use anchor_lang::prelude::*;

declare_id!("AcCKUb5GBSGgX1dz6GMTyvZwSBsnnzZJigac9kB7cfXr"); // 类函数宏:声明程序 ID

#[program] // 属性宏:定义 Solana 程序
pub mod macro_example {
    use super::*;

    pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
        Ok(())
    }
}

#[derive(Accounts)] // 自定义派生宏:为结构体生成账户验证逻辑
pub struct Initialize {}

这些宏的具体工作原理将在后续文章中详细讲解。


【笔记配套代码】 https://github.com/0xE1337/rareskills_evm_to_solana 【参考资料】 https://learnblockchain.cn/column/119 https://www.rareskills.io/solana-tutorial

点赞 0
收藏 0
分享
本文参与登链社区写作激励计划 ,好文好收益,欢迎正在阅读的你也加入。

0 条评论

请先 登录 后评论