Rust之过程式的宏(一)

Rust的宏有两种类型:声明式和程序式。声明式的宏很像C语言的宏,而过程式的宏很像Java有反射加持的AOP。
而程序式根据作用的位置(顶在谁头上)不同,分成自定义derive函数属性三种。
在当前目录下创建三个工程:

crate new use_demo
crate new --lib hello_macro
crate new --lib hello_macro_derive

hello_macro包

lib.rs:

pub trait HelloMacro {`在这里插入代码片`
    fn hello_macro();
}

hello_macro_derive包

lib.rs:

extern crate proc_macro;

use proc_macro::TokenStream;
use quote::quote;
use syn;

#[proc_macro_derive(HelloMacro)]
pub fn hello_macro_derive(input: TokenStream) -> TokenStream {
    // Construct a representation of Rust code as a syntax tree
    // that we can manipulate
    let ast = syn::parse(input).unwrap();

    // Build the trait implementation
    impl_hello_macro(&ast)
}

fn impl_hello_macro(ast: &syn::DeriveInput) -> TokenStream {
    let name = &ast.ident;
    let gen = quote! {
        impl HelloMacro for #name {
            fn hello_macro() {
                println!("Hello, Macro! My name is {}!", stringify!(#name));
            }
        }
    };
    gen.into()
}

Cargo.toml

[lib]
proc-macro = true

[dependencies]
syn = "1.0"
quote = "1.0"

use_macro包

main.rs

use hello_macro::HelloMacro;
use hello_macro_derive::HelloMacro;

#[derive(HelloMacro)]
struct Pancakes;

fn main() {
    Pancakes::hello_macro();
}

Cargo.toml

[dependencies]
hello_macro = { path = "../hello_macro" }
hello_macro_derive = { path = "../hello_macro/hello_macro_derive" }

运行结果

$ cargo run
Compiling proc-macro2 v0.4.30
Compiling unicode-xid v0.1.0
Compiling hello_macro v0.1.0 (G:\workspace_rh\bbc\study2\hello_macro)
Compiling quote v0.6.13
Compiling syn v0.14.9
Compiling hello_macro_derive v0.1.0 (G:\workspace_rh\bbc\study2\hello_macro\hello_macro_derive)
Compiling use_macro v0.1.0 (G:\workspace_rh\bbc\study2\use_macro)
Finished dev [unoptimized + debuginfo] target(s) in 8.67s
Running target\debug\use_macro.exe
Hello, Macro! My name is Pancakes

总结

  1. 世界上本没有宏,有了proc-macro = true它(Trait)就变成了宏。原本Trait是一个安安静静的Trait,是#[proc_macro_derive(HelloMacro)]把它和一个专门用来做宏动作(宏定义)的lib.rs绑到了一起。
  2. proc-macro = true必须放到宏定义所在的package。因为:the #[proc_macro_derive] attribute is only usable with crates of the proc-macro crate type
  3. 定义宏的lib.rs,只能定义宏,而不能有其它的item。至少目前是这样( proc-macro crate types currently cannot export any items other than fun ctions tagged with #[proc_macro], #[proc_macro_derive], or #[proc_macro_att ribute])
  4. 这就是为什么,一些开源比如Diesel,把别人(第三方)的Trait,通过自己增加宏定义的方式使用它。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值