练习题来自:https://practice-zh.course.rs/pattern-match/match-iflet.html
1
rust
// 填空
enum Direction {
East,
West,
North,
South,
}
fn main() {
let dire = Direction::South;
match dire {
Direction::East => println!("East"),
__ => { // 在这里匹配 South 或 North
println!("South or North");
},
_ => println!(__),
};
}
答案:
rust
fn main() {
let dire = Direction::South;
match dire {
Direction::East => println!("East"),
Direction::North | Direction::South => {
println!("South or North");
},
_ => println!("West"),
};
}
2 🌟🌟 match 是一个表达式,因此可以用在赋值语句中
rust
fn main() {
let boolean = true;
// 使用 match 表达式填空,并满足以下条件
//
// boolean = true => binary = 1
// boolean = false => binary = 0
let binary = __;
assert_eq!(binary, 1);
}
答案
rust
fn main() {
let boolean = true;
let binary = match boolean {
true => 1,
false => 2,
};
assert_eq!(binary, 1);
}
3 🌟🌟 使用 match 匹配出枚举成员持有的值
rust
// 填空
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::Move
assert_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++实现类似的功能,我觉得应该需要多态的辅助。不过本质上将多个不同类型放在一起匹配的意义也很有限。
rust
fn show_message(msg: Message) {
match msg {
Message::Move { x:a, y:b } => { // 这里匹配 Message::Move
assert_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
rust
fn main() {
let alphabets = ['a', 'E', 'Z', '0', 'x', '9' , 'Y'];
// 使用 `matches` 填空
for ab in alphabets {
assert!(__)
}
}
这里的matches!
类似于将match
的条件仅限制为一个,成功与否返回布尔值。
rust
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
rust
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不支持重载基本符号,加个语法糖也行啊。
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
rust
fn main() {
let o = Some(7);
// 移除整个 `match` 语句块,使用 `if let` 替代
match o {
Some(i) => {
println!("This is a really long string and `{:?}`", i);
}
_ => {}
};
}
这种if
语法有点奇怪,按理说if
的条件判断都应该用双等于号才对,但是理解为赋值就好了。
rust
fn main() {
let o = Some(7);
if let Some(i) = o {
println!("This is a really long string and `{:?}`", i);
}
}
7
rust
// 填空
enum Foo {
Bar(u8)
}
fn main() {
let a = Foo::Bar(1);
__ {
println!("foobar 持有的值是: {}", i);
}
}
由于Foo
中只有一个枚举值,if
匹配百分百会命中。
rust
fn main() {
let a = Foo::Bar(1);
if let Foo::Bar(i) = a {
println!("foobar 持有的值是: {}", i);
}
}
8
rust
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
匹配。
rust
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
rust
// 就地修复错误
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),
_ => ()
}
}
答案:
rust
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),
_ => ()
}
}