Rust指南枚舉類(lèi)與模式匹配詳解
前言
書(shū)接上文,本篇博客分享的是Rust 枚舉類(lèi)與模式匹配 的知識(shí)。作為安全性強(qiáng)的語(yǔ)言,Rust 的枚舉類(lèi)并不像其他編程語(yǔ)言中的概念那樣簡(jiǎn)單,但依然可以十分簡(jiǎn)單的使用。
1、Rust基本枚舉類(lèi)語(yǔ)法
枚舉允許我們列舉所有可能的值來(lái)定義一個(gè)類(lèi)型,枚舉中的值也叫變體
1.1、定義枚舉
我們舉一個(gè)例子:
IP地址:IPV4、IPV6
enum IpAddrKind{ V4, V6 }
接收枚舉值
let four=IpAddrKind::V4; let six=IpAddrKind::V6;
枚舉的變體都位于標(biāo)識(shí)符的命名空間下,使用::
進(jìn)行分割
1.2、將數(shù)據(jù)附加到枚舉的變體中
形式如下:
enum IpAddr{ V4(String), V6(String) }
優(yōu)點(diǎn):
- 不需要額外使用
struct
來(lái)確定類(lèi)型 - 每個(gè)變體可以擁有不同的類(lèi)型以及關(guān)聯(lián)的數(shù)據(jù)量
例如:
#[derive(Debug)] enum IpAddrKind { V4(u8,u8,u8,u8), V6(String) } fn main() { let home=IpAddrKind::V4(127, 0, 0, 1); let loopback=IpAddrKind::V6(String::from("這是IPV6")); println!("{:?}\n{:?}",home,loopback); }
運(yùn)行效果:
#[derive(Debug)]
作為Rust提供的調(diào)試庫(kù)是可以直接輸出結(jié)構(gòu)體和枚舉類(lèi)型的- 但是注意占位符只能使用
{:?}
- 標(biāo)準(zhǔn)庫(kù)中的IpAddr
struct IpV4Addr{ //--snip-- } struct IpV6Addr{ //--snip-- } enum IpAddr { V4(IpV4Addr), V6(IpV6Addr) }
1.3、變體的多種嵌套方式
enum Message { Quit, Move {x:i32,y:u32}, Write(String), ChangeColor(i32,i32,i32) } fn main() { let q=Message::Quit; let m=Message::Move { x: 6, y: 12 }; let w=Message::Write(String::from("hello_world")); let c=Message::ChangeColor(255, 255, 0); }
在這段代碼中枚舉類(lèi)變體一共有四種數(shù)據(jù)類(lèi)型:
- 不帶關(guān)聯(lián)數(shù)據(jù)
Quit
- 匿名結(jié)構(gòu)體
Move
- 字符串類(lèi)型
Write
- 匿名元組結(jié)構(gòu)體
ChangeColor
1.4、定義枚舉方法
和結(jié)構(gòu)體方法類(lèi)似,使用impl
關(guān)鍵字:
impl Message{ fn call(&self){} }
這里就不具體實(shí)現(xiàn)了,此時(shí)枚舉的所有變體都可以調(diào)用call
方法,例如q.call();
2、Option枚舉
2.1、引入Option枚舉解決控制問(wèn)題
Option
是 Rust 標(biāo)準(zhǔn)庫(kù)中的枚舉類(lèi),這個(gè)類(lèi)用于填補(bǔ) Rust 不支持null
引用的空白。- 許多語(yǔ)言支持 null 的存在(C/C++、Java),這樣很方便,但也制造了極大的問(wèn)題,null 的發(fā)明者也承認(rèn)這一點(diǎn),“一個(gè)方便的想法造成累計(jì) 10 億美元的損失”。
null
經(jīng)常在開(kāi)發(fā)者把一切都當(dāng)作不是 null 的時(shí)候給予程序致命一擊:畢竟只要出現(xiàn)一個(gè)這樣的錯(cuò)誤,程序的運(yùn)行就要徹底終止。- 為了解決這個(gè)問(wèn)題,很多語(yǔ)言默認(rèn)不允許 null,但在語(yǔ)言層面支持 null 的出現(xiàn)(常在類(lèi)型前面用 ? 符號(hào)修飾)。
- Java 默認(rèn)支持 null,但可以通過(guò) @NotNull 注解限制出現(xiàn) null,這是一種應(yīng)付的辦法。
Rust 在語(yǔ)言層面徹底不允許空值 null 的存在,但無(wú)奈null 可以高效地解決少量的問(wèn)題,所以 Rust 引入了 Option 枚舉類(lèi):
enum Option<T>{ Some(T), None }
2.2、枚舉類(lèi)的具體使用
枚舉類(lèi)包含在預(yù)導(dǎo)入模塊中(Prelude),可直接使用:
let some_number=Some(5); let some_string=Some("a string") let absent:Option<&str>=None;
注意:
- 編譯器無(wú)法推斷None是什么類(lèi)型,所以一定要顯示聲明
- 由于
absent
屬于None的變體,因此是無(wú)效數(shù)據(jù),也就是null
3、match控制流運(yùn)算符
- 枚舉的目的是對(duì)某一類(lèi)事物的分類(lèi),分類(lèi)的目的是為了對(duì)不同的情況進(jìn)行描述。
- 基于這個(gè)原理,往往枚舉類(lèi)最終都會(huì)被分支結(jié)構(gòu)處理(許多語(yǔ)言中的 switch )。
- switch 語(yǔ)法很經(jīng)典,但在 Rust 中并不支持,很多語(yǔ)言摒棄 switch 的原因都是因?yàn)?switch 容易存在因忘記添加 break 而產(chǎn)生的串接運(yùn)行問(wèn)題,Java 和 C# 這類(lèi)語(yǔ)言通過(guò)安全檢查杜絕這種情況出現(xiàn)。
Rust 通過(guò) match 語(yǔ)句來(lái)實(shí)現(xiàn)分支結(jié)構(gòu)。先認(rèn)識(shí)一下如何用 match 處理枚舉類(lèi):
fn main() { enum Book { Papery {index: u32}, Electronic {url: String}, } let book = Book::Papery{index: 1001}; let ebook = Book::Electronic{url: String::from("url...")}; match book { Book::Papery { index } => { println!("Papery book {}", index); }, Book::Electronic { url } => { println!("E-book {}", url); } } } //運(yùn)行結(jié)果:Papery book 1001
這是由于book
屬于Papery
的變體,因此會(huì)執(zhí)行第一個(gè)打印語(yǔ)句
match 塊也可以當(dāng)作函數(shù)表達(dá)式來(lái)對(duì)待,它也是可以有返回值的:
match 枚舉類(lèi)實(shí)例 { 分類(lèi)1 => 返回值表達(dá)式, 分類(lèi)2 => 返回值表達(dá)式, ... }
但是要謹(jǐn)記:所有返回值表達(dá)式的類(lèi)型必須一樣!
如果把枚舉類(lèi)附加屬性定義成元組,在 match 塊中需要臨時(shí)指定一個(gè)名字:
enum Book { Papery(u32), Electronic {url: String}, } let book = Book::Papery(1001); match book { Book::Papery(i) => { println!("{}", i); }, Book::Electronic { url } => { println!("{}", url); } }
變體Papery指定了i
變量,Electronic指定了url
match 除了能夠?qū)γ杜e類(lèi)進(jìn)行分支選擇以外,還可以對(duì)整數(shù)、浮點(diǎn)數(shù)、字符和字符串切片引用(&str)類(lèi)型的數(shù)據(jù)進(jìn)行分支選擇。其中,浮點(diǎn)數(shù)類(lèi)型被分支選擇雖然合法,但不推薦這樣使用,因?yàn)榫葐?wèn)題可能會(huì)導(dǎo)致分支錯(cuò)誤。
對(duì)非枚舉類(lèi)進(jìn)行分支選擇時(shí)必須注意處理例外情況,即使在例外情況下沒(méi)有任何要做的事。例外情況用下劃線 _ 表示:
fn main() { let t = "abc"; match t { "abc" => println!("Yes"), _ => {}, } }
4、if let 語(yǔ)法
通過(guò)一個(gè)簡(jiǎn)單的流程控制代碼理解此部分知識(shí):
let i = 0; match i { 0 => println!("zero"), _ => {}, } //主函數(shù)中運(yùn)行結(jié)果:zero
這段程序的目的是判斷 i 是否是數(shù)字 0,如果是就打印 zero。
那么現(xiàn)在用 if let 語(yǔ)法縮短這段代碼:
let i = 0; if let 0 = i { println!("zero"); }
if let 語(yǔ)法格式如下:
if let 匹配值 = 源變量 { 語(yǔ)句塊 }
- 可以在之后添加一個(gè)
else
塊來(lái)處理例外情況。
if let 語(yǔ)法可以認(rèn)為是只區(qū)分兩種情況的 match 語(yǔ)句的"語(yǔ)法糖"
在枚舉類(lèi)中的使用:
fn main() { enum Book { Papery(u32), Electronic(String) } let book = Book::Electronic(String::from("url")); if let Book::Papery(index) = book { println!("Papery {}", index); } else { println!("Not papery book"); } } //運(yùn)行結(jié)果:Not papery book
Rust 枚舉類(lèi)和模式匹配的知識(shí)就分享到這里了,期待你的鼓勵(lì),這將是我創(chuàng)作的不竭動(dòng)力!
到此這篇關(guān)于Rust指南枚舉類(lèi)與模式匹配精講的文章就介紹到這了,更多相關(guān)Rust枚舉類(lèi)與模式匹配內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Rust-使用dotenvy加載和使用環(huán)境變量的過(guò)程詳解
系統(tǒng)的開(kāi)發(fā),測(cè)試和部署離不開(kāi)環(huán)境變量,今天分享在Rust的系統(tǒng)開(kāi)發(fā)中,使用dotenvy來(lái)讀取和使用環(huán)境變量,感興趣的朋友跟隨小編一起看看吧2023-11-11