Rust Refutability(可反駁性): 模式是否會(huì)匹配失效

2023-03-22 15:15 更新
ch18-02-refutability.md
commit d44317c3122b44fb713aba66cc295dee3453b24b

模式有兩種形式:refutable(可反駁的)和 irrefutable(不可反駁的)。能匹配任何傳遞的可能值的模式被稱為是 不可反駁的irrefutable)。一個(gè)例子就是 let x = 5; 語句中的 x,因?yàn)?nbsp;x 可以匹配任何值所以不可能會(huì)失敗。對(duì)某些可能的值進(jìn)行匹配會(huì)失敗的模式被稱為是 可反駁的refutable)。一個(gè)這樣的例子便是 if let Some(x) = a_value 表達(dá)式中的 Some(x);如果變量 a_value 中的值是 None 而不是 Some,那么 Some(x) 模式不能匹配。

函數(shù)參數(shù)、 let 語句和 for 循環(huán)只能接受不可反駁的模式,因?yàn)橥ㄟ^不匹配的值程序無法進(jìn)行有意義的工作。if let 和 while let 表達(dá)式被限制為只能接受可反駁的模式,因?yàn)楦鶕?jù)定義他們意在處理可能的失?。簵l件表達(dá)式的功能就是根據(jù)成功或失敗執(zhí)行不同的操作。

通常我們無需擔(dān)心可反駁和不可反駁模式的區(qū)別,不過確實(shí)需要熟悉可反駁性的概念,這樣當(dāng)在錯(cuò)誤信息中看到時(shí)就知道如何應(yīng)對(duì)。遇到這些情況,根據(jù)代碼行為的意圖,需要修改模式或者使用模式的結(jié)構(gòu)。

讓我們看看一個(gè)嘗試在 Rust 要求不可反駁模式的地方使用可反駁模式以及相反情況的例子。在示例 18-8 中,有一個(gè) let 語句,不過模式被指定為可反駁模式 Some(x)。如你所見,這不能編譯:

    let Some(x) = some_option_value;

示例 18-8: 嘗試在 let 中使用可反駁模式

如果 some_option_value 的值是 None,其不會(huì)成功匹配模式 Some(x),表明這個(gè)模式是可反駁的。然而,因?yàn)?nbsp;let 對(duì)于 None 匹配不能產(chǎn)生任何合法的代碼,所以 let 語句只能接受不可反駁模式。Rust 會(huì)在編譯時(shí)抱怨我們嘗試在要求不可反駁模式的地方使用可反駁模式:

$ cargo run
   Compiling patterns v0.1.0 (file:///projects/patterns)
error[E0005]: refutable pattern in local binding: `None` not covered
   --> src/main.rs:3:9
    |
3   |     let Some(x) = some_option_value;
    |         ^^^^^^^ pattern `None` not covered
    |
    = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
    = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
    = note: the matched value is of type `Option<i32>`
help: you might want to use `if let` to ignore the variant that isn't matched
    |
3   |     if let Some(x) = some_option_value { /* */ }
    |

For more information about this error, try `rustc --explain E0005`.
error: could not compile `patterns` due to previous error

因?yàn)槲覀儧]有覆蓋(也不可能覆蓋?。┑侥J?nbsp;Some(x) 的每一個(gè)可能的值,所以 Rust 會(huì)合理地抗議。

為了修復(fù)在需要不可反駁模式的地方使用可反駁模式的情況,可以修改使用模式的代碼:不同于使用 let,可以使用 if let。如此,如果模式不匹配,大括號(hào)中的代碼將被忽略,其余代碼保持有效。示例 18-9 展示了如何修復(fù)示例 18-8 中的代碼。

    if let Some(x) = some_option_value {
        println!("{}", x);
    }

示例 18-9: 使用 if let 和一個(gè)帶有可反駁模式的代碼塊來代替 let

我們給了代碼一個(gè)得以繼續(xù)的出路!雖然我們沒辦法在避免產(chǎn)生錯(cuò)誤的情況下使用不可反駁模式,但這段使用可反駁模式的代碼是完全有效的。如果為 if let 提供了一個(gè)總是會(huì)匹配的模式,比如示例 18-10 中的 x,編譯器會(huì)給出一個(gè)警告:

    if let x = 5 {
        println!("{}", x);
    };

示例 18-10: 嘗試把不可反駁模式用到 if let 上

Rust 會(huì)抱怨將不可反駁模式用于 if let 是沒有意義的:

$ cargo run
   Compiling patterns v0.1.0 (file:///projects/patterns)
warning: irrefutable `if let` pattern
 --> src/main.rs:2:8
  |
2 |     if let x = 5 {
  |        ^^^^^^^^^
  |
  = note: `#[warn(irrefutable_let_patterns)]` on by default
  = note: this pattern will always match, so the `if let` is useless
  = help: consider replacing the `if let` with a `let`

warning: `patterns` (bin "patterns") generated 1 warning
    Finished dev [unoptimized + debuginfo] target(s) in 0.39s
     Running `target/debug/patterns`
5

基于此,match匹配分支必須使用可反駁模式,除了最后一個(gè)分支需要使用能匹配任何剩余值的不可反駁模式。Rust允許我們?cè)谥挥幸粋€(gè)匹配分支的match中使用不可反駁模式,不過這么做不是特別有用,并可以被更簡(jiǎn)單的 let 語句替代。

目前我們已經(jīng)討論了所有可以使用模式的地方,以及可反駁模式與不可反駁模式的區(qū)別,下面讓我們一起去把可以用來創(chuàng)建模式的語法過目一遍吧。


以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)