练习题来自:https://practice-zh.course.rs/pattern-match/match-iflet.html
1
// 填空
enum Direction {East,West,North,South,
}fn main() {let dire = Direction::South;match dire {Direction::East => println!("East"),__ => { // 在这里匹配 South 或 Northprintln!("South or North");},_ => println!(__),};
}
答案:
fn main() {let dire = Direction::South;match dire {Direction::East => println!("East"),Direction::North | Direction::South => {println!("South or North");},_ => println!("West"),};
}
2 🌟🌟 match 是一个表达式,因此可以用在赋值语句中
fn main() {let boolean = true;// 使用 match 表达式填空,并满足以下条件//// boolean = true => binary = 1// boolean = false => binary = 0let binary = __;assert_eq!(binary, 1);
}
答案
fn main() {let boolean = true;let binary = match boolean {true => 1,false => 2,};assert_eq!(binary, 1);
}
3 🌟🌟 使用 match 匹配出枚举成员持有的值
// 填空
enum Message {Quit,Move { x: i32, y: i32 },Write(String),ChangeColor(i32, i32, i32),
}fn main() {let msgs = [Message::Quit,Message::Move{x:1, y:3},Message::ChangeColor(255,255,0)];for msg in msgs {show_message(msg)}
} fn show_message(msg: Message) {match msg {__ => { // 这里匹配 Message::Moveassert_eq!(a, 1);assert_eq!(b, 3);},Message::ChangeColor(_, g, b) => {assert_eq!(g, __);assert_eq!(b, __);}__ => println!("no data in these variants")}
}
枚举的加入真是把match
完全玩活了,和其他语言的switch case
完全不一样了。C++实现类似的功能,我觉得应该需要多态的辅助。不过本质上将多个不同类型放在一起匹配的意义也很有限。
fn show_message(msg: Message) {match msg {Message::Move { x:a, y:b } => { // 这里匹配 Message::Moveassert_eq!(a, 1);assert_eq!(b, 3);},Message::ChangeColor(_, g, b) => {assert_eq!(g, 255);assert_eq!(b, 0);}__ => println!("no data in these variants")}
}
4
fn main() {let alphabets = ['a', 'E', 'Z', '0', 'x', '9' , 'Y'];// 使用 `matches` 填空for ab in alphabets {assert!(__)}
}
这里的matches!
类似于将match
的条件仅限制为一个,成功与否返回布尔值。
fn main() {let alphabets = ['a', 'E', 'Z', '0', 'x', '9' , 'Y'];// 使用 `matches` 填空for ab in alphabets {assert!(matches!(ab, 'A'..='Z' | 'a'..='z'| '0'..='9'))}
}
5
enum MyEnum {Foo,Bar
}fn main() {let mut count = 0;let v = vec![MyEnum::Foo,MyEnum::Bar,MyEnum::Foo];for e in v {if e == MyEnum::Foo { // 修复错误,只能修改本行代码count += 1;}}assert_eq!(count, 2);
}
这里展示出来的matches!
又类似于等号对枚举的重载,这次我站C++,明显是等号更直观,就算Rust不支持重载基本符号,加个语法糖也行啊。
fn main() {let mut count = 0;let v = vec![MyEnum::Foo,MyEnum::Bar,MyEnum::Foo];for e in v {if matches!(e, MyEnum::Foo) { // 修复错误,只能修改本行代码count += 1;}}assert_eq!(count, 2);
}
6
fn main() {let o = Some(7);// 移除整个 `match` 语句块,使用 `if let` 替代match o {Some(i) => {println!("This is a really long string and `{:?}`", i);}_ => {}};
}
这种if
语法有点奇怪,按理说if
的条件判断都应该用双等于号才对,但是理解为赋值就好了。
fn main() {let o = Some(7);if let Some(i) = o {println!("This is a really long string and `{:?}`", i);}
}
7
// 填空
enum Foo {Bar(u8)
}fn main() {let a = Foo::Bar(1);__ {println!("foobar 持有的值是: {}", i);}
}
由于Foo
中只有一个枚举值,if
匹配百分百会命中。
fn main() {let a = Foo::Bar(1);if let Foo::Bar(i) = a {println!("foobar 持有的值是: {}", i);}
}
8
enum Foo {Bar,Baz,Qux(u32)
}fn main() {let a = Foo::Qux(10);// 移除以下代码,使用 `match` 代替if let Foo::Bar = a {println!("match foo::bar")} else if let Foo::Baz = a {println!("match foo::baz")} else {println!("match others")}
}
使用match
时,注意default
匹配。
fn main() {let a = Foo::Qux(10);match a {Foo::Bar => println!("match foo::bar"),Foo::Baz => println!("match foo::baz"),_ => println!("match others"),}
}
题外话,如果你使用IDE自带的format对代码进行格式化,任何列表匹配的最后一项都会自动加一个顿号。C++的枚举类也支持这个功能,但其他的语言结构和其他语言我就不清楚了。这么设计的原因是,git对代码变更统计是以行为单位的,如果最后一个元素不加顿号,那么新增一个元素时,必然要在最后补一个顿号,这就造成了两行代码变更,会让审查代码的人很困惑。
9
// 就地修复错误
fn main() {let age = Some(30);if let Some(age) = age { // 创建一个新的变量,该变量与之前的 `age` 变量同名assert_eq!(age, Some(30));} // 新的 `age` 变量在这里超出作用域match age {// `match` 也能实现变量遮蔽Some(age) => println!("age 是一个新的变量,它的值是 {}",age),_ => ()}}
答案:
fn main() {let age = Some(30);if let Some(age) = age { // 创建一个新的变量,该变量与之前的 `age` 变量同名assert_eq!(age, 30);} // 新的 `age` 变量在这里超出作用域match age {// `match` 也能实现变量遮蔽Some(age) => println!("age 是一个新的变量,它的值是 {}",age),_ => ()}}