欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 教育 > 锐评 > 【Rust练习】15.match 和 if let

【Rust练习】15.match 和 if let

2024/10/25 6:24:53 来源:https://blog.csdn.net/qq_37387199/article/details/142288121  浏览:    关键词:【Rust练习】15.match 和 if let

练习题来自: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),_ => ()}}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com