match 是什么语法?PHP8 也加了

polarisxu

共 7096字,需浏览 15分钟

 ·

2021-06-11 13:55

阅读本文大概需要 5 分钟。

大家好,我是站长 polarisxu。

这是 Rust 劝退系列的第 8 个教程,探讨 Rust 中的模式匹配。

01 match 表达式

关于 match 表达式,很多其他语言并没有,比如 Go 语言。不过有些语言开始支持 match,比如 PHP 8.0 就有了 match 表达式。

一般地可以认为 match 和 switch 类似,所以 Rust 中没有 switch。

match 用于检查某个当前的值是否匹配一组/列值中的某一个。看一个具体的例子:

fn test_match(number: i32) -> &'static str {
    match number {
        // 匹配单个值
        1 => {println!("One!"); "One!"},
        // 匹配多个值
        2 | 3 | 5 | 7 | 11 => "This is a prime",
        // 匹配一个闭区间范围
        13..=19 => "A teen",
        // 处理其他情况
        _ => "Ain't special",
    }
}

看起来是一个简单的语法结构,但大概率在其他语言没见过。简单解释下:

  • 跟其他语言的 switch 类似,可以匹配多个分支;多个分支之间,使用 , 分隔;
  • 在 match 分支中,=> 左侧是模式,因此叫做模式匹配,比如 | 表示匹配多个值;..= 表示匹配一个范围;右侧是在左侧匹配成功时要执行的操作;
  • match 要求穷尽,也就是要包含所有可能的值。因此提供了 _,用来处理所有其他情况,类似 switch 的 default 分支;但只要穷尽了,可以没有 _
  • 如果右侧操作是多个语句,需要放在 {} 中;
  • match 是表达式,它的结果是匹配到的模式中,执行操作的最后一个表达式的结果。这在 Rust 中是很常见的,之前提到过,Rust 中一切皆表达式。所以,这个例子中 match 表达式的值即为函数的返回值。因此,match 的所有分支必须返回同一数据类型;
  • 注意 match 表达式最后是否有分号的区别;

日常吐槽:在 match 中匹配区间,如果想和 for in 一样,使用 .. 来表示半闭半开区间,结果报错。看到资料说应该使用 ,但却提示该语法已废弃!为啥语法结构还不保持一致呢?!

看一个接收 match 结果的例子:

let boolean = true;
let binary = match boolean {
  false => 0,
  true => 1,
}; // 注意这里的分号
println!("{} -> {}", boolean, binary);

02 match 其他用法

上面介绍了常规的 match 操作。match 还有很多其他的用法。

解构

当元组和 match 一起时,可以解构元组。

fn main() {
  // 试一试将不同的值赋给 `pair`
    let pair = (0, -2);
    
    println!("Tell me about {:?}", pair);
    // match 可以解构一个元组
    match pair {
        // 解构出第二个值
        (0, y) => println!("First is `0` and `y` is `{:?}`", y),
        (x, 0) => println!("`x` is `{:?}` and last is `0`", x),
        _      => println!("It doesn't matter what they are"),
        // `_` 表示不将值绑定到变量
    }
}

关于枚举和指针、引用和 match 的结合,以后遇到再讲解。

guard 语句

在 match 分支中可以加上过滤条件。接着上面元组解构的例子:

fn main() {
    let pair = (2, -2);

    println!("Tell me about {:?}", pair);
    match pair {
        (x, y) if x == y => println!("These are twins"),
        // `if` 条件部分是一个卫语句
        (x, y) if x + y == 0 => println!("Antimatter, kaboom!"),
        (x, _) if x % 2 == 1 => println!("The first one is odd"),
        _ => println!("No correlation..."),
    }
}

绑定

这是什么意思呢?看一个例子:(来自 rust by example)

// `age` 函数,返回一个 `u32` 值。
fn age() -> u32 {
    15
}

fn main() {
    println!("Tell me type of person you are");

    match age() {
        0             => println!("I'm not born yet I guess"),
        // 可以直接 `match` 1 ..= 12,但怎么把岁数打印出来呢?
        // 在 1 ..= 12 分支中绑定匹配值到 `n` 。现在年龄就可以读取了。
        n @ 1  ..= 12 => println!("I'm a child of age {:?}", n),
        n @ 13 ..= 19 => println!("I'm a teen of age {:?}", n),
        // 不符合上面的范围。返回结果。
        n             => println!("I'm an old person of age {:?}", n),
    }
}

match 后是一个函数,我们希望在分支中,根据匹配结果,使用 age 函数的返回值。当然,这个例子有点多此一举,完全可以在 match 之前用变量存储 age 函数的返回值。

那换一个例子:

fn some_number() -> Option<u32> {
    Some(41)
}

fn main() {
    match some_number() {
        Some(n @ 40..=42) => println!("The Answer: {}!", n),
        Some(n)      => println!("Not interesting... {}", n),
        _            => (),
    }
}
  • 关于 Option 以后讲解

这个例子很好的讲解了绑定的作用:分支中想要使用匹配的结果,通过 @ 符号可以将匹配的结果和某个变量绑定,然后就可以使用这个变量了。

03 if let 和 while let

这两个结构其他语言没见过,可以理解为是某些场景下替代 match,让代码更简洁,因为 match 必须穷尽所有情况,而 if let 和 while let 没有此限制。

以下代码:

let some_u8_value = Some(3u8);
match some_u8_value {
  Some(3) => println!("three"),
  _ => (), // 有点多余
}

改为 if let:

let some_u8_value = Some(3u8);
if let Some(3) = some_u8_value {
 println!("three");
}
  • 和 match 一样,if let 和 while let 都是表达式;
  • if/while let 等号左侧是模式,右侧是要匹配的值;所以当右侧的值和左侧的模式匹配时,执行对应的语句块;所以,有时候 if let 也可以单纯的当做解构使用;
  • if let 支持普通的 else if 和 else;while let 没有 else;

while let 语法和 if let 类似。这里就不举例子了。

04 小结

Rust 中的 match 虽然和其他语言的 switch 类似,很显然,match 的复杂度比 switch 高。当然,不管复杂与否,最关键还是要实际使用,需要不断实际练习。

通常,match 和 Option、枚举一起使用,因此,在讲解这两个知识点时,一般会使用到 match。




往期推荐


我是 polarisxu,北大硕士毕业,曾在 360 等知名互联网公司工作,10多年技术研发与架构经验!2012 年接触 Go 语言并创建了 Go 语言中文网!著有《Go语言编程之旅》、开源图书《Go语言标准库》等。


坚持输出技术(包括 Go、Rust 等技术)、职场心得和创业感悟!欢迎关注「polarisxu」一起成长!也欢迎加我微信好友交流:gopherstudio


浏览 29
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报