欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文弄懂rust聲明宏

 更新時(shí)間:2024年03月31日 11:27:17   作者:又耳筆記  
Rust支持兩種宏,一種是聲明宏,一種是過程宏,本文主要介紹了一文弄懂rust聲明宏,通過聲明宏可以減少一些樣板代碼,具有一定的參考價(jià)值,感興趣的可以了解一下

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中的方法與關(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
  • Rust指南之生命周期機(jī)制詳解

    Rust指南之生命周期機(jī)制詳解

    Rust?生命周期機(jī)制是與所有權(quán)機(jī)制同等重要的資源管理機(jī)制,之所以引入這個(gè)概念主要是應(yīng)對(duì)復(fù)雜類型系統(tǒng)中資源管理的問題,這篇文章主要介紹了Rust指南之生命周期機(jī)制詳解,需要的朋友可以參考下
    2022-10-10
  • Rust中的內(nèi)部可變性與RefCell<T>詳解

    Rust中的內(nèi)部可變性與RefCell<T>詳解

    內(nèi)部可變性允許在不可變引用中修改內(nèi)部數(shù)據(jù),通過RefCell在運(yùn)行時(shí)檢查借用規(guī)則,適用于Mock對(duì)象和多所有權(quán)的可變性場(chǎng)景,結(jié)合Rc和RefCell實(shí)現(xiàn)多所有者共享并修改數(shù)據(jù),但僅適用于單線程
    2025-02-02
  • 如何使用VSCode配置Rust開發(fā)環(huán)境(Rust新手教程)

    如何使用VSCode配置Rust開發(fā)環(huán)境(Rust新手教程)

    這篇文章主要介紹了如何使用VSCode配置Rust開發(fā)環(huán)境(Rust新手教程),本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • rust中的match表達(dá)式使用詳解

    rust中的match表達(dá)式使用詳解

    在rust中提供了一個(gè)極為強(qiáng)大的控制流運(yùn)算符match,這篇文章主要介紹了rust中的match表達(dá)式,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-08-08
  • Rust 原始指針功能探索

    Rust 原始指針功能探索

    這篇文章主要為大家介紹了Rust 原始指針功能探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10
  • C和Java沒那么香了,Serverless時(shí)代Rust即將稱王?

    C和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
  • rust中trait的使用方法詳解

    rust中trait的使用方法詳解

    trait用中文來講就是特征,它就是一個(gè)標(biāo)記,只不過這個(gè)標(biāo)記被用在特定的地方,也就是類型參數(shù)的后面,下面我們就來學(xué)習(xí)一下trait的具體使用方法吧
    2023-12-12
  • Rust標(biāo)量類型的具體使用

    Rust標(biāo)量類型的具體使用

    本文主要介紹了Rust標(biāo)量類型的具體使用,其中包括整數(shù)類型、浮點(diǎn)類型、布爾類型以及字符類型,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-03-03
  • Rust指南枚舉類與模式匹配詳解

    Rust指南枚舉類與模式匹配詳解

    這篇文章主要介紹了Rust指南枚舉類與模式匹配精講,枚舉允許我們列舉所有可能的值來定義一個(gè)類型,枚舉中的值也叫變體,今天通過一個(gè)例子給大家詳細(xì)講解,需要的朋友可以參考下
    2022-09-09

最新評(píng)論