美文网首页Rust
关于rust宏-过程宏(补充)

关于rust宏-过程宏(补充)

作者: 神奇的考拉 | 来源:发表于2022-04-28 17:34 被阅读0次

一、 rust编译过程

rust编译过程
从上面的编译过程图,可以看到声明宏过程宏被编译到AST中过程是不同的:
  • 声明宏:通过macro_rule 定义的宏最终只是被解析为TokenStream;
  • 过程宏: 定义的过程宏首先被解析为proc_macro::TokenStream,接着再被解析为proc_macro2::TokenStream,再使用Syn库解析为AST(这里的不同于编译生成的AST),最后再通过Quote解析为TokenStream,最终会被扩展到编译过程TokenStream中,进而再被编译AST;

二、过程宏

1、样例


属性过程宏样例

此处定义一个属性过程宏的样例
2、定义过程宏

  • 准备工作
    主要是Cargo.toml文件中
[lib]
proc-macro=true # 开启过程宏

# 引入如下三个crate的原因 参考上面编译过程图:在生成最终的TokenStream前,需要依赖这三个crate
[dependencies]
proc-macro2 = "1.0.34"
syn = {version="1.0.82", feature="full"}
quote ="1.0.10"
  • 项目结构
    过程宏需要定义在一个单独的crate中,主要是因为过程宏是一段在编译crate前,对其代码进行加工的代码,而这段是需要在编译后执行的。若是将定义过程宏和使用过程宏放到同一个crate中,就会陷入编译“死锁”:
    1、要编译的代码需要运行过程宏进行展开,否则代码是不完整的,无法编译crate;
    2、不能编译crate,那么在crate中的过程宏就无法执行,就不能展开被过程宏“修饰”的代码

  • 定义过程宏

use proc_macro::TokenStream;

/// 定义一个属性过程宏
/// ???输入参数:TokenStream 输出参数: TokenStream【解释见`项目结构`】
/// 不做任何加工,直接输出item
#[proc_macro_attribute]
pub fn my_first_attr_proc_macro(attr: TokenStream, item: TokenStream) -> TokenStream {
    eprintln!("===输出attr");
    eprintln!(" {:#?}", attr);
    eprintln!("===输出item");
    eprintln!("{:#?}", item);
    item
}
  • 使用过程宏
    1、Cargo.toml文件引入过程宏crate
[dependencies]
produceral_macro={path= "../procedural_macro_demos/procedural_macro" } # 本地路径引用

2、使用过程宏
通过#[crate_name::proc_macro_func("过程宏名字-任意")]
类似 #[produceral_macro::my_first_attr_proc_macro("my_first_procedural_macro")]

#[produceral_macro::my_first_attr_proc_macro("my_first_procedural_macro")]
fn add(a: u64, b: u64) -> () {
    eprintln!("a={:?}, b={:?}, a+b={:?}", a, b , (a+b));
}

rust过程宏本质就是一个编译环节的“过滤器”或者说是一个“中间件”,接收一段用户编写的源代码,再做一通“转换”操作,然后返回给编译器一段经过修改的代码。

目前过程宏代码的调试,最好通过print来进行

最终生成的TokenStream是没有任何语义信息的,是通过树形结构的数据组织,表达了用户源代码各个语言元素的类型及其相互之间的关系;每个语言元素都有一个span属性,记录该元素在用户源代码中的位置。同时不同类型的节点有各自独有的属性。
而Rust过程宏就是自己能够手动修改输入变量中的值,比如样例中的attr,item,换句话说等价于加工原始输入代码,最后将加工后的代码返回给编译器即可.

三、proc-macro2\syn\quote三个包在过程宏中的应用

使用syn、quote等crate模拟过程宏“修改”源代码,并将结果以TokenStream给到编译器:

#[proc_macro_attribute]
pub fn my_first_attr_proc_macro_parse(attr: TokenStream, input: TokenStream) -> TokenStream {
    eprintln!("===attr: {:#?}", attr);
    eprintln!("===item: {:#?}", input);
    // 通过syn来解析输入的TokenStream
    let input_fn = syn::parse_macro_input!(input as syn::ItemFn);
    // 获取该token对应的类型:ident
    let name = input_fn.sig.ident.clone();
    //  输出TokenStream给到编译器
    TokenStream::from(quote! { // 使用quote库来构建编译器需要的TokenStream
        fn #name () {
            #input_fn

            for i in 0..3 {
                println!("loop time {}", i);

                let r = std::panic::catch_unwind(|| {
                    #name();
                });

                if r.is_ok() {
                    return;
                }
                if i == 2 {
                    std::panic::resume_unwind(r.unwrap_err());
                }
            }
        }
    })
}

测试代码

#[procedural_macro::my_first_attr_proc_macro_parse]
fn test_proc_macro() {
    assert_eq!(1,1);  // 运行正常
    assert_eq!(1,22);  // 触发panic
}

四、编译过程宏代码

#输出未进行宏扩展的ast树
$ cargo rustc -- -Z ast-json-noexpand=yes
#输出宏扩展后的ast树
$ cargo rustc -- -Z ast-json=yes 
#输出hir格式的中间描述
$ cargo rustc -- -Z unpretty=hir 
#输出hir格式并带有类型信息的中间描述
$ cargo rustc -- -Z unpretty=hir,typed 
#输出hir格式并带有完整树结构的中间描述
$ cargo rustc -- -Z unpretty=hir-tree
#输出mir格式的中间描述
$ cargo rustc -- -Z unpretty=mir

#### 代码编译中间代码 ####
#输出llvm ir格式的中间描述
$ rustc --emit llvm-ir lrfrc.rs
#输出汇编格式的中间描述
$ rustc --emit asm lrfrc.rs
#分析中间汇编输出
$ rustc -C opt-level=3 --emit=obj lrfrc.rs
$ size -A lrfrc.o
$ objdump -d lrfrc.o

相关文章

  • 关于rust宏-过程宏(补充)

    一、 rust编译过程 从上面的编译过程图,可以看到 和 被编译到AST中过程是不同的: 声明宏:通过macro_...

  • 关于rust宏补充(二)-派生过程宏示例

    rust中过程宏示例: 准备工作 定义一个过程宏: proc_macro_derive 测试 更多例子[https...

  • rust - macro_rules! 过程宏学习笔记 macr

    rust中宏大致分两种:过程宏: 形如 println!(), vec!() 这类属性宏: 形如 #[deriv...

  • 关于rust的“宏”

    一、概述 为了解决rust语法元素的扩展,并能复用现有的代码,在rust编写的程序中普遍使用宏.通过宏定义和宏调用...

  • Rust 过程宏之derive派生宏入门

    需求: 获取枚举类型附加信息,如备注说明,传统作法如下: 实现 comment 方法根据匹配值返回对应字符串 传统...

  • Rust语言编程实例100题-048

    Rust语言编程实例100题-048 题目:Rust 对宏(macro)有着非常好的支持。宏能够使得你能够通过写代...

  • 宏(Macro)指的是 Rust 中一系列的功能 声明(Declarative)宏, 使用macro_rules!...

  • vs code 配置rust开发环境

    实现的需求: 查看宏定义,代码补全, 安装这三个插件: Rust,Rust Test Lens,rust-anal...

  • rust--宏

  • Rust 的宏

    首先, 我感觉 Rust 的宏是非常强大的, 可以根据 match 有不同的实现,但是它也是极其让人,起码是让我纠...

网友评论

    本文标题:关于rust宏-过程宏(补充)

    本文链接:https://www.haomeiwen.com/subject/dkjdyrtx.html