# 匹配
> [match.md](https://github.com/rust-lang/rust/blob/master/src/doc/book/match.md)
commit fc4bb5f77060b5822f25edbabbdf7a1d48a7f8fe
一个简单的[`if`](#)/`else`往往是不够的,因为你可能有两个或更多个选项。这样`else`也会变得异常复杂。Rust 有一个`match`关键字,它可以让你有效的取代复杂的`if`/`else`组。看看下面的代码:
~~~
let x = 5;
match x {
1 => println!("one"),
2 => println!("two"),
3 => println!("three"),
4 => println!("four"),
5 => println!("five"),
_ => println!("something else"),
}
~~~
`match`使用一个表达式然后基于它的值分支。每个分支都是`val => expression`这种形式。当匹配到一个分支,它的表达式将被执行。`match`属于“模式匹配”的范畴,`match`是它的一个实现。有[一个整个关于模式的部分](#)讲到了所有可能的模式。
那么这有什么巨大的优势呢?这确实有优势。第一,`match`强制*穷尽性检查*(*exhaustiveness checking*)。你看到了最后那个下划线开头的分支了吗?如果去掉它,Rust 将会给我们一个错误:
~~~
error: non-exhaustive patterns: `_` not covered
~~~
Rust 试图告诉我们忘记了一个值。编译器从`x`推断它可以是任何正的 32 位整型值;例如从 1 到 2,147,483,647。`_`就像一个*匹配所有*的分支,它会捕获所有没有被`match`分支捕获的所有可能值。如你所见,在上个例子中,我们提供了 1 到 5 的`mtach`分支,如果`x`是 6 或者其他值,那么它会被`_`捕获。
`match`也是一个表达式,也就是说它可以用在`let`绑定的右侧或者其它直接用到表达式的地方:
~~~
let x = 5;
let number = match x {
1 => "one",
2 => "two",
3 => "three",
4 => "four",
5 => "five",
_ => "something else",
};
~~~
有时,这是一个把一种类型的数据转换为另一个类型的好方法。
### 匹配枚举(Matching on enums)
`match`的另一个重要的作用是处理枚举的可能变量:
~~~
enum Message {
Quit,
ChangeColor(i32, i32, i32),
Move { x: i32, y: i32 },
Write(String),
}
fn quit() { /* ... */ }
fn change_color(r: i32, g: i32, b: i32) { /* ... */ }
fn move_cursor(x: i32, y: i32) { /* ... */ }
fn process_message(msg: Message) {
match msg {
Message::Quit => quit(),
Message::ChangeColor(r, g, b) => change_color(r, g, b),
Message::Move { x: x, y: y } => move_cursor(x, y),
Message::Write(s) => println!("{}", s),
};
}
~~~
再一次,Rust编译器检查穷尽性,所以它要求对每一个枚举的变量都有一个匹配分支。如果你忽略了一个,除非你用`_`否则它会给你一个编译时错误。
与之前的`match`的作用不同,你不能用常规的`if`语句来做这些。你可以使用[if let](#)语句,它可以被看作是一个`match`的简略形式。
- 前言
- 贡献者
- 1.介绍
- 2.准备
- 3.学习 Rust
- 3.1.猜猜看
- 3.2.哲学家就餐问题
- 3.3.其它语言中的 Rust
- 4.语法和语义
- 4.1.变量绑定
- 4.2.函数
- 4.3.原生类型
- 4.4.注释
- 4.5.If语句
- 4.6.循环
- 4.7.所有权
- 4.8.引用和借用
- 4.9.生命周期
- 4.10.可变性
- 4.11.结构体
- 4.12.枚举
- 4.13.匹配
- 4.14.模式
- 4.15.方法语法
- 4.16.Vectors
- 4.17.字符串
- 4.18.泛型
- 4.19.Traits
- 4.20.Drop
- 4.21.if let
- 4.22.trait 对象
- 4.23.闭包
- 4.24.通用函数调用语法
- 4.25.crate 和模块
- 4.26.const和static
- 4.27.属性
- 4.28.type别名
- 4.29.类型转换
- 4.30.关联类型
- 4.31.不定长类型
- 4.32.运算符和重载
- 4.33.Deref强制多态
- 4.34.宏
- 4.35.裸指针
- 4.36.不安全代码
- 5.高效 Rust
- 5.1.栈和堆
- 5.2.测试
- 5.3.条件编译
- 5.4.文档
- 5.5.迭代器
- 5.6.并发
- 5.7.错误处理
- 5.8.选择你的保证
- 5.9.外部函数接口
- 5.10.Borrow 和 AsRef
- 5.11.发布途径
- 5.12.不使用标准库
- 6.Rust 开发版
- 6.1.编译器插件
- 6.2.内联汇编
- 6.4.固有功能
- 6.5.语言项
- 6.6.链接进阶
- 6.7.基准测试
- 6.8.装箱语法和模式
- 6.9.切片模式
- 6.10.关联常量
- 6.11.自定义内存分配器
- 7.词汇表
- 8.语法索引
- 9.参考文献
- 附录:名词中英文对照