一文弄懂rust聲明宏
Rust支持兩種宏,一種是聲明宏,一種是過程宏,前者相較于后者還是比較簡(jiǎn)單的。本文主要是講解Rust元編程里的聲明宏,通過聲明宏可以減少一些樣板代碼,它是一個(gè)用代碼生成代碼的技術(shù)。
聲明宏的主要原理是通過匹配傳入的代碼然后替換成指定的代碼,因?yàn)樘鎿Q是發(fā)生在編譯器,所以rust的宏編程沒有任何運(yùn)行時(shí)的開銷,可以放心的用,不用擔(dān)心性能 :)。
快速入門
聲明宏不像過程宏那樣需要在單獨(dú)的包(package/crate)中定義,只需要使用macro_rules!就可以簡(jiǎn)單的定義一個(gè)聲明宏,一個(gè)簡(jiǎn)單的示例如下。
// https://youerning.top/post/rust-declarative-macros-tutorial/ macro_rules! add { ($a:expr, $b:expr) => { $a + $b }; } fn main() { let sum = add!(1,2); println!("sum: {sum}"); }
輸出如下:
sum: 3
上面這個(gè)結(jié)果應(yīng)該不會(huì)讓人意外,你會(huì)發(fā)現(xiàn)聲明宏定義的那一段代碼和普通的match代碼非常相似,不同的在于變量前面多了個(gè)前綴$, 而且需要通過冒號(hào):注明變量的類型,這里的變量類型是expr,這是表達(dá)式的意思。
聲明宏語法
一個(gè)聲明宏大致可以分為三個(gè)部分
- 聲明宏的名稱定義,比如例子中的add
- 模式匹配部分, 比如例子中的($a:expr, $b:expr)
- 聲明宏返回的部分, 也就是花括號(hào)被包裹的部分, 比如例子中的$a + $b
本文的開頭說過,過程宏的原理就是通過匹配傳入的代碼然后替換成指定的代碼, 所以上面的例子在編譯(展開)之后應(yīng)該會(huì)變成下面的代碼。
fn main() { let sum = 1 + 2; println!("sum: {sum}"); }
如果我們傳遞三個(gè)參數(shù)呢? 比如add!(1,2,3),那么它會(huì)在編譯的時(shí)候報(bào)以下錯(cuò)誤。
error: no rules expected the token `,`
--> src\main.rs:8:23
|
1 | macro_rules! add {
| ---------------- when calling this macro
...
8 | let sum = add!(1,2,3);
| ^ no rules expected this token in macro call
|
note: while trying to match meta-variable `$b:expr`
--> src\main.rs:2:15
|
2 | ($a:expr, $b:expr)=>{
| ^^^^^^^error: could not compile `declarative-macros` (bin "declarative-macros") due to previous error
其實(shí)這很好理解,我們的模式只能匹配兩個(gè)變量$a和$b, 但是add!(1,2,3)卻傳入了三個(gè)變量,所以匹配不了,那么就會(huì)報(bào)錯(cuò),因?yàn)檫@是不合法的語法。
那么,怎么匹配三個(gè)變量,或者是一個(gè)變量呢? 有兩個(gè)辦法,一是一一對(duì)應(yīng),二是使用重復(fù)的匹配方法。為了簡(jiǎn)單起見,我們先使用比較笨的方法,代碼如下。
macro_rules! add { // 聲明宏的第一條匹配規(guī)則 ($a: expr) => { $a }; // 聲明宏的第二條匹配規(guī)則 ($a:expr, $b:expr)=>{ $a + $b }; // 聲明宏的第三條匹配規(guī)則 ($a:expr, $b:expr, $c: expr)=>{ $a + $b }; } fn main() { let sum = add!(1); println!("sum1: {sum}"); let sum = add!(1,2); println!("sum2: {sum}"); let sum = add!(1,2,3); println!("sum3: {sum}"); }
上面的代碼和快速入門的例子沒有太大的區(qū)別,主要的區(qū)別是之前的例子只有一個(gè)匹配規(guī)則,而新的例子有三條匹配規(guī)則,當(dāng)rust編譯代碼的時(shí)候,會(huì)將調(diào)用聲明宏的輸入?yún)?shù)從上至下依次匹配每條規(guī)則,當(dāng)匹配到就會(huì)停止匹配,然后返回對(duì)應(yīng)的代碼,這和rust的match模式匹配沒有太大的區(qū)別,唯一的區(qū)別可能是, 聲明宏使用;分隔不同的匹配模式,而match的不同匹配模式使用,分隔。
上面的代碼輸出如下:
sum1: 1
sum2: 3
sum3: 3
這樣的結(jié)果并不讓人意外,唯一讓人沮喪的是,每種情況都寫一個(gè)對(duì)應(yīng)的表達(dá)式的話,得累死去。
元變量
現(xiàn)在讓我們繼續(xù)看看rust的聲明宏支持哪些類型。
- item: 條目,比如函數(shù)、結(jié)構(gòu)體、模組等。
- block: 區(qū)塊(即由花括號(hào)包起的一些語句加上/或是一項(xiàng)表達(dá)式)。
- stmt: 語句
- pat: 模式
- expr: 表達(dá)式
- ty: 類型
- ident: 標(biāo)識(shí)符
- path: 路徑 (例如 foo, ::std::mem::replace, transmute::<_, int>, …)
- meta: 元條目,即被包含在 #[...]及#![...]屬性內(nèi)的東西。
- tt: 標(biāo)記樹
大多數(shù)情況,一般只會(huì)使用expr和tt, 使用expr是因?yàn)閞ust中幾乎可以被稱為基于表達(dá)式的編程語言,因?yàn)樗谋磉_(dá)式概念非常大,即使是if和while這樣的語句也可以作為一個(gè)表達(dá)式返回值,而tt是一個(gè)萬金油,它可以簡(jiǎn)單的被認(rèn)為是其他類型都不匹配的情況下的兜底類型。
下面看一個(gè)tt類型的例子。
macro_rules! add { ($a: tt) => { { println!("{}", stringify!($a)); 1 } }; } fn main() { let sum = add!(1); println!("sum: {sum}"); let sum = add!(,); println!("sum: {sum}"); let sum = add!({}); println!("sum: {sum}"); let sum = add!(youerning); println!("sum: {sum}"); }
代碼輸出如下:
1
sum: 1
,
sum: 1
{}
sum: 1
youerning
sum: 1
代碼展開后長(zhǎng)這樣:
值得注意的是: 下面的代碼是手動(dòng)的展開,與真實(shí)的編譯代碼還是有點(diǎn)區(qū)別的!!!
fn main() { let sum = { println!("{}", "1") 1 }; println!("sum: {sum}"); let sum = { println!("{}", ",") 1 }; println!("sum: {sum}"); let sum = { println!("{}", "{}") 1 }; println!("sum: {sum}"); }
總的來說, tt這個(gè)類型可以接受合法或者不合法的各種標(biāo)識(shí)符。
stringify!是啥? 說實(shí)話我也不太懂,我的理解是,你可以將任何東西扔給它,它會(huì)返回一個(gè)字符串字面量給你。
宏展開(expand)
如果我真的能夠手動(dòng)展開自己的代碼,那就肯定會(huì)了,也就不用開文章學(xué)習(xí)了不是,所以如果吃不準(zhǔn)宏展開之后的結(jié)果或者故障排查的時(shí)候可以使用cargo expand命令查看展開后的代碼。
可以通過以下命令安裝。
cargo install cargo-expand
安裝之后在項(xiàng)目的根目錄執(zhí)行cargo expand即可,上面的例子展開之后如下。
#![feature(prelude_import)] #[prelude_import] use std::prelude::rust_2021::*; #[macro_use] extern crate std; fn main() { let sum = { { ::std::io::_print(format_args!("{0}\n", "1")); }; 1 }; { ::std::io::_print(format_args!("sum: {0}\n", sum)); }; let sum = { { ::std::io::_print(format_args!("{0}\n", ",")); }; 1 }; { ::std::io::_print(format_args!("sum: {0}\n", sum)); }; let sum = { { ::std::io::_print(format_args!("{0}\n", "{}")); }; 1 }; { ::std::io::_print(format_args!("sum: {0}\n", sum)); }; let sum = { { ::std::io::_print(format_args!("{0}\n", "youerning")); }; 1 }; { ::std::io::_print(format_args!("sum: {0}\n", sum)); }; }
如果看不太懂可以結(jié)合我手動(dòng)展開的代碼一起看。
標(biāo)記樹撕咬機(jī)(TT muncher)
通過標(biāo)記樹撕咬機(jī)(TT muncher)我們可以實(shí)現(xiàn)遞歸的聲明宏,不過在此之前讓我們先解決不定參數(shù)的問題,之前解決的方案是根據(jù)要傳的參數(shù)編寫聲明宏的匹配代碼,這樣實(shí)在是太不優(yōu)雅了,讓我們看看怎么一次性搞定。
macro_rules! add { ($($a: expr),*) => { 0$(+$a)* }; } fn main() { let sum = add!(); println!("sum1: {sum}"); let sum = add!(1); println!("sum1: {sum}"); let sum = add!(1,2); println!("sum2: {sum}"); let sum = add!(1,2,3); println!("sum3: {sum}"); }
輸出如下:
sum1: 0
sum1: 1
sum2: 3
sum3: 6
重復(fù)
聲明宏里面有一些難點(diǎn),其中一個(gè)就是重復(fù)的匹配模式, 也就是這個(gè)例子中的$($a: expr),*, 為啥要這樣寫? 因?yàn)檫@是rust的語法, 就像定義一個(gè)新變量必須使用let表達(dá)式一樣,這個(gè)不需要太糾結(jié)。
下面來看看這種模式的語法定義,重復(fù)的一般形式是$ ( ... ) sep rep
- $ 是字面標(biāo)記。
- ( ... ) 代表了將要被重復(fù)匹配的模式,由小括號(hào)包圍。
- sep是一個(gè)可選的分隔標(biāo)記。常用例子包括,和;。
- rep是重復(fù)控制標(biāo)記。當(dāng)前有兩種選擇,分別是* (代表接受0或多次重復(fù))以及+ (代表1或多次重復(fù))。目前沒有辦法指定“0或1”或者任何其它更加具體的重復(fù)計(jì)數(shù)或區(qū)間。
大家可以將($($a: expr),*)改成($($a: expr);*),然后就會(huì)發(fā)現(xiàn)編譯不過了,因?yàn)榉指舴枰?了
也就是說, $($a: expr),*匹配到了(), (1), (1,2),(1,2,3),為啥能匹配到()?, 因?yàn)?能匹配0個(gè)或多個(gè),所以零參數(shù)的()也能匹配上,如果你將這個(gè)例子中的*換成+,就會(huì)發(fā)現(xiàn)add!()會(huì)報(bào)錯(cuò),因?yàn)?要求至少一個(gè)參數(shù)。
下面以參數(shù)(1,2,3)的例子再深入一下宏展開時(shí)的操作,當(dāng)傳入(1,2,3)時(shí),因?yàn)楦?($a: expr),*能夠匹配上, 所以(1,2,3)里的冒號(hào),被$($a: expr),*的冒號(hào),給匹配上,而$a代表1 2 3中的每個(gè)元素, 那么怎么在返回的代碼中標(biāo)識(shí)重復(fù)的參數(shù)呢?rust的語法是, 我們需要使用$()*將$a包裹起來,外面的包裝代碼對(duì)應(yīng)參數(shù)匹配時(shí)的重復(fù)次數(shù), 你可以簡(jiǎn)單的將$()*認(rèn)為是必要的語法。
下面看一個(gè)簡(jiǎn)單的例子
macro_rules! print { ($($a: expr),*) => { println!("{} {}", $($a),*) }; } fn main() { print!(1,2); }
$($a),*會(huì)原封不動(dòng)的將參數(shù)放在它對(duì)應(yīng)的位置,因?yàn)閜rintln!指定了兩個(gè)位置參數(shù),所以使用自定義的print只能傳遞兩個(gè)參數(shù)。
最后看看上面那個(gè)add!宏的例子, add!(1,2,3)展開之后應(yīng)該變成下面這樣。
0+1+2+3
之所以這樣,是因?yàn)槲覀冊(cè)诜祷氐拇a模式中$($a)*在$a前面加了一個(gè)+, 而這個(gè)加號(hào)+因?yàn)楸?()*包裹,所以會(huì)跟著$a重復(fù)一樣的次數(shù),也就變成了+1+2+3。
為啥前面要加個(gè)0?因?yàn)椴患?的話, 就不是合法的表達(dá)式了。
遞歸示例1
雖然add!這個(gè)宏可以使用一個(gè)模式匹配就能完成,但是我們可以使用更加復(fù)雜的方式實(shí)現(xiàn),也就是標(biāo)記樹撕咬機(jī)(TT muncher)。
macro_rules! add { ($a: expr) => { $a }; ($a: expr, $b: expr) => { $a + $b }; ($a: expr, $($other: tt)*) => { $a + add!($($other)*) }; } fn main() { let sum = add!(1,2,3,4,5); println!("sum: {sum}"); }
使用**標(biāo)記樹撕咬機(jī)(TT muncher)**的代碼和之前的代碼結(jié)果沒有什么區(qū)別,但是展開的過程中會(huì)有些不同,因?yàn)楹笳呤褂昧诉f歸,它的遞歸調(diào)用類似于add!(1, add!(2, add!(3, add!(3, add!(3, add!(5))))));
這段代碼的前兩個(gè)匹配模式不用過多介紹,關(guān)鍵在于最后一個(gè)($a: expr, $($other: tt)*), $a 和 ,會(huì)吃掉一個(gè)參數(shù)和一個(gè)逗號(hào),, 而$($other: tt)*會(huì)匹配到后面所有的參數(shù)2,3,4,5。
注意這些參數(shù)包含逗號(hào),, 還有就是我們?cè)谑褂?($other: tt)*這種重復(fù)模式的時(shí)候沒有指定分隔符, 所以tt既匹配了參數(shù)2 3 4 5也匹配了分割這些數(shù)字的逗號(hào),, 所以在展開的代碼$a + add!($($other)*)會(huì)變成1 + add!(2,3,4,5), 然后就是不斷的遞歸了,直到遇到第一個(gè)匹配模式。
遞歸示例2
你可能在上一個(gè)例子不能感受到**標(biāo)記樹撕咬機(jī)(TT muncher)**的威力,所以我們繼續(xù)看下一個(gè)例子。
我們可以通過**標(biāo)記樹撕咬機(jī)(TT muncher)**的遞歸調(diào)用來生成對(duì)嵌套對(duì)象的遞歸調(diào)用,這樣就不需要不斷的判斷Option的值是Some還是None了。
use serde_json::{json, Value}; macro_rules! serde_get { ($value: ident, $first: expr) => { { match ($value).get($first) { Some(val) => Some(val), None => { None } } } }; ($value: ident, $first: expr, $($others:expr),+) => { { match ($value).get($first) { Some(val) => { serde_get!(val, $($others),+) }, None => { None } } } }; ($value: ident, $first: expr, $($others:tt)* ) => { { match ($ident).get($first) { Some(val) => { serde_get!(val, $($others)+), } None => None } } }; } fn main() { let object = json!({ "key11": {"key12": "key13"}, "key21": {"key22": {"key23": "key24"}} }); if let Some(val) = serde_get!(object, "xx") { println!(r#"object["a"]["b"]["c"]={val:?}"#); } else { println!(r#"object["a"]["b"]["c"]不存在"#); } if let Some(val) = serde_get!(object, "key1", "key12") { println!(r#"object["key11"]["key12"] = {val:}"#); } if let Some(val) = serde_get!(object, "key21", "key22", "key23") { println!(r#"object["key21"]["key21"]["key23"] = {val:}"#); } }
這個(gè)例子寫完,我才發(fā)現(xiàn)serde_json可以直接使用["key21"]["key21"]["key23"]這樣的語法直接判斷!!!, 不過serde_json的返回結(jié)果都是null, 如果鍵值對(duì)不存在的話。
總結(jié)
我感覺rust的宏編程還是很有意思的,不過這東西的確得真正有需求的時(shí)候才會(huì)真的理解,我之前也不是太懂,看了視頻和文章也不是太懂,只是知道它能干啥,但是沒有一個(gè)真正要解決的問題,所以一直不能很好的掌握,直到在使用serde_json時(shí)遇到嵌套的數(shù)據(jù)結(jié)構(gòu)需要寫重復(fù)的判斷代碼時(shí),我才在應(yīng)用的時(shí)候掌握了聲明宏(雖然最后發(fā)現(xiàn)它的實(shí)用價(jià)值可能不是那么大),至于過程宏,可能等我遇到需要過程宏的時(shí)候才會(huì)很好的掌握吧,到時(shí)候在寫對(duì)應(yīng)的文章吧。
參考鏈接
https://earthly.dev/blog/rust-macros/
https://doc.rust-lang.org/reference/macros-by-example.html#metavariables
https://www.bookstack.cn/read/DaseinPhaos-tlborm-chinese/mbe-macro-rules.md
https://veykril.github.io/tlborm/
https://github.com/dtolnay/cargo-expandhttps://youerning.top/post/rust/rust-declarative-macros-tutorial/
到此這篇關(guān)于一文弄懂rust聲明宏的文章就介紹到這了,更多相關(guān)rust聲明宏內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Rust中的方法與關(guān)聯(lián)函數(shù)使用解讀
在Rust中,方法是定義在特定類型(如struct)的impl塊中,第一個(gè)參數(shù)是self(可變或不可變),方法用于描述該類型實(shí)例的行為,而關(guān)聯(lián)函數(shù)則不包含self參數(shù),常用于構(gòu)造新實(shí)例或提供一些與實(shí)例無關(guān)的功能,Rust的自動(dòng)引用和解引用特性使得方法調(diào)用更加簡(jiǎn)潔2025-02-02如何使用VSCode配置Rust開發(fā)環(huán)境(Rust新手教程)
這篇文章主要介紹了如何使用VSCode配置Rust開發(fā)環(huán)境(Rust新手教程),本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07C和Java沒那么香了,Serverless時(shí)代Rust即將稱王?
Serverless Computing,即”無服務(wù)器計(jì)算”,其實(shí)這一概念在剛剛提出的時(shí)候并沒有獲得太多的關(guān)注,直到2014年AWS Lambda這一里程碑式的產(chǎn)品出現(xiàn)。Serverless算是正式走進(jìn)了云計(jì)算的舞臺(tái)2021-06-06