Refutabilidade: Se um Padrão Pode Falhar ao Corresponder
Os padrões vêm em duas formas: refutáveis e irrefutáveis. Padrões que
correspondem a qualquer valor possível passado são irrefutáveis. Um exemplo
é x na instrução let x = 5;, porque x corresponde a qualquer coisa e,
portanto, não pode falhar. Já os padrões que podem falhar ao corresponder a
algum valor possível são refutáveis. Um exemplo é Some(x) na expressão
if let Some(x) = a_value, porque, se o valor na variável a_value for
None em vez de Some, o padrão Some(x) não corresponderá.
Parâmetros de função, instruções let e loops for só podem aceitar padrões
irrefutáveis, porque o programa não pode fazer nada significativo quando os
valores não correspondem. Já as expressões if let e while let, bem como a
instrução let...else, aceitam padrões refutáveis e irrefutáveis, mas o
compilador alerta contra padrões irrefutáveis porque, por definição, essas
construções servem para lidar com possíveis falhas: a utilidade de uma
condicional está em sua capacidade de se comportar de maneira diferente
dependendo do sucesso ou do fracasso.
Em geral, você não deveria se preocupar com a distinção entre padrões refutáveis e irrefutáveis; no entanto, precisa estar familiarizado com o conceito de refutabilidade para conseguir responder quando o vir em uma mensagem de erro. Nesses casos, você precisará alterar o padrão ou a construção com a qual está usando o padrão, dependendo do comportamento pretendido do código.
Vejamos um exemplo do que acontece quando tentamos usar um padrão refutável onde
Rust exige um padrão irrefutável, e vice-versa. A Listagem 19-8 mostra uma
instrução let, mas, no padrão, especificamos Some(x), um padrão refutável.
Como você pode imaginar, esse código não vai compilar.
fn main() {
let some_option_value: Option<i32> = None;
let Some(x) = some_option_value;
}
letSe some_option_value fosse um valor None, ele deixaria de corresponder ao
padrão Some(x), o que significa que o padrão é refutável. No entanto, a
instrução let só aceita um padrão irrefutável, porque não existe nada válido
que o código possa fazer com um valor None. Em tempo de compilação, Rust
reclamará que tentamos usar um padrão refutável onde um padrão irrefutável é
necessário:
$ cargo run
Compiling patterns v0.1.0 (file:///projects/patterns)
error[E0005]: refutable pattern in local binding
--> 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/ch19-02-refutability.html
= note: the matched value is of type `Option<i32>`
help: you might want to use `let else` to handle the variant that isn't matched
|
3 | let Some(x) = some_option_value else { todo!() };
| ++++++++++++++++
For more information about this error, try `rustc --explain E0005`.
error: could not compile `patterns` (bin "patterns") due to 1 previous error
Porque não cobrimos (e não poderíamos cobrir!) todos os valores válidos com o
padrão Some(x), Rust produz corretamente um erro do compilador.
Se tivermos um padrão refutável onde é necessário um padrão irrefutável,
podemos corrigir isso mudando o código que usa o padrão: em vez de usar let,
podemos usar let...else. Então, se o padrão não corresponder, o código entre
chaves tratará o valor. A Listagem 19-9 mostra como corrigir o código da
Listagem 19-8.
fn main() {
let some_option_value: Option<i32> = None;
let Some(x) = some_option_value else {
return;
};
}
let...else e um bloco com padrões refutáveis no lugar de letDemos uma saída para o código! Esse código é perfeitamente válido, embora isso
signifique que não podemos usar um padrão irrefutável sem receber um aviso. Se
passarmos a let...else um padrão que sempre corresponderá, como x, conforme
mostrado na Listagem 19-10, o compilador emitirá um aviso.
fn main() {
let x = 5 else {
return;
};
}
let...elseRust reclama que não faz sentido usar let...else com um
padrão irrefutável:
$ cargo run
Compiling patterns v0.1.0 (file:///projects/patterns)
warning: irrefutable `let...else` pattern
--> src/main.rs:2:5
|
2 | let x = 5 else {
| ^^^^^^^^^
|
= note: this pattern will always match, so the `else` clause is useless
= help: consider removing the `else` clause
= note: `#[warn(irrefutable_let_patterns)]` on by default
warning: `patterns` (bin "patterns") generated 1 warning
Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.39s
Running `target/debug/patterns`
Por essa razão, os braços de match devem usar padrões refutáveis, exceto o
último braço, que deve corresponder a quaisquer valores restantes com um padrão
irrefutável. Rust nos permite usar um padrão irrefutável em um match com
apenas um braço, mas essa sintaxe não é particularmente útil e pode ser
substituída por uma instrução let mais simples.
Agora que você sabe onde usar padrões e conhece a diferença entre padrões refutáveis e irrefutáveis, vamos cobrir toda a sintaxe que podemos usar para criar padrões.