当前位置:   article > 正文

rust枚举类和可以为None值的枚举对象

rust枚举类和可以为None值的枚举对象

枚举类在 Rust 中并不像其他编程语言中的概念那样简单,但依然可以十分简单的使用:

  1. #[derive(Debug)]
  2. enum Book {
  3.     Papery, Electronic
  4. }
  5. fn main() {
  6.     let book = Book::Papery;
  7.     println!("{:?}", book);
  8. }

运行结果:

书分为纸质书(Papery book)和电子书(Electronic book)

如果你现在正在开发一个图书管理系统,你需要描述两种书的不同属性(纸质书有索书号,电子书只有 URL),你可以为枚举类成员添加元组属性描述:

  1. enum Book {
  2. Papery(u32),
  3. Electronic(String),
  4. }
  5. let book = Book::Papery(1001);
  6. let ebook = Book::Electronic(String::from("url://..."));

如果你想为属性命名,可以用结构体语法:

  1. enum Book {
  2. Papery { index: u32 },
  3. Electronic { url: String },
  4. }
  5. let book = Book::Papery{index: 1001};

虽然可以如此命名,但请注意,并不能像访问结构体字段一样访问枚举类绑定的属性。访问的方法在 match 语法中。

match 语法

枚举的目的是对某一类事物的分类,分类的目的是为了对不同的情况进行描述。基于这个原理,往往枚举类最终都会被分支结构处理(许多语言中的 switch )。 switch 语法很经典,但在 Rust 中并不支持,很多语言摒弃 switch 的原因都是因为 switch 容易存在因忘记添加 break 而产生的串接运行问题,Java 和 C# 这类语言通过安全检查杜绝这种情况出现。

Rust 通过 match 语句来实现分支结构。先认识一下如何用 match 处理枚举类:

  1. fn main() {
  2.     enum Book {
  3.         Papery {index: u32},
  4.         Electronic {url: String},
  5.     }
  6.    
  7.     let book = Book::Papery{index: 1001};
  8.     let ebook = Book::Electronic{url: String::from("url...")};
  9.    
  10.     match book {
  11.         Book::Papery { index } => {
  12.             println!("Papery book {}", index);
  13.         },
  14.         Book::Electronic { url } => {
  15.             println!("E-book {}", url);
  16.         }
  17.     }
  18. }

运行结果:

Papery book 1001

match 块也可以当作函数表达式来对待,它也是可以有返回值的:

  1. match 枚举类实例 {
  2. 分类1 => 返回值表达式,
  3. 分类2 => 返回值表达式,
  4. ...
  5. }

但是所有返回值表达式的类型必须一样!

如果把枚举类附加属性定义成元组,在 match 块中需要临时指定一个名字:

  1. enum Book {
  2.     Papery(u32),
  3.     Electronic {url: String},
  4. }
  5. let book = Book::Papery(1001);
  6. match book {
  7.     Book::Papery(i) => {
  8.         println!("{}", i);
  9.     },
  10.     Book::Electronic { url } => {
  11.         println!("{}", url);
  12.     }
  13. }

 match 除了能够对枚举类进行分支选择以外,还可以对整数、浮点数、字符和字符串切片引用(&str)类型的数据进行分支选择。其中,浮点数类型被分支选择虽然合法,但不推荐这样使用,因为精度问题可能会导致分支错误。

对非枚举类进行分支选择时必须注意处理例外情况,即使在例外情况下没有任何要做的事 . 例外情况用下划线 _ 表示:

  1. fn main() {
  2.     let t = "abc";
  3.     match t {
  4.         "abc" => println!("Yes"),
  5.         _ => {},
  6.     }
  7. }

 

Option 枚举类

Option 是 Rust 标准库中的枚举类,这个类用于填补 Rust 不支持 null 引用的空白。

许多语言支持 null 的存在(C/C++、Java),这样很方便,但也制造了极大的问题,null 的发明者也承认这一点,"一个方便的想法造成累计 10 亿美元的损失"。

null 经常在开发者把一切都当作不是 null 的时候给予程序致命一击:毕竟只要出现一个这样的错误,程序的运行就要彻底终止。

为了解决这个问题,很多语言默认不允许 null,但在语言层面支持 null 的出现(常在类型前面用 ? 符号修饰)。

Java 默认支持 null,但可以通过 @NotNull 注解限制出现 null,这是一种应付的办法。

Rust 在语言层面彻底不允许空值 null 的存在,但无奈null 可以高效地解决少量的问题,所以 Rust 引入了 Option 枚举类:

  1. enum Option<T> {
  2. Some(T),
  3. None,
  4. }

 如果你想定义一个可以为空值的类,你可以这样:

let opt = Option::Some("Hello");

如果你想针对 opt 执行某些操作,你必须先判断它是否是 Option::None

  1. fn main() {
  2.     let opt = Option::Some("Hello");
  3.     match opt {
  4.         Option::Some(something) => {
  5.             println!("{}", something);
  6.         },
  7.         Option::None => {
  8.             println!("opt is nothing");
  9.         }
  10.     }
  11. }

运行结果:Hello

如果你的变量刚开始是空值,你体谅一下编译器,它怎么知道值不为空的时候变量是什么类型的呢?

所以初始值为空的 Option 必须明确类型:

  1. fn main() {
  2.     let opt: Option<&str> = Option::None;
  3.     match opt {
  4.         Option::Some(something) => {
  5.             println!("{}", something);
  6.         },
  7.         Option::None => {
  8.             println!("opt is nothing");
  9.         }
  10.     }
  11. }

运行结果:

opt is nothing

这种设计会让空值编程变得不容易,但这正是构建一个稳定高效的系统所需要的。由于 Option 是 Rust 编译器默认引入的,在使用时可以省略 Option:: 直接写 None 或者 Some()。

  1. // 可以为None和Some类型的枚举
  2. let mut nfc_timer: Option<i32> = None;
  3. nfc_timer = Some(111);
  4. println!("nfc_timer value is {:?}", nfc_timer);
  5. match nfc_timer {
  6. None => {}
  7. Some(_) => {
  8. println!("nfc_timer has value: {:?}", nfc_timer);
  9. }
  10. }

Option 是一种特殊的枚举类,它可以含值分支选择:

  1. fn main() {
  2.         let t = Some(64);
  3.         match t {
  4.                 Some(64) => println!("Yes"),
  5.                 _ => println!("No"),
  6.         }
  7. }

if let 语法

  1. let i = 0;
  2. match i {
  3.     0 => println!("zero"),
  4.     _ => {},
  5. }

 

放入主函数运行结果:zero

这段程序的目的是判断 i 是否是数字 0,如果是就打印 zero。

现在用 if let 语法缩短这段代码:

  1. let i = 0;
  2. if let 0 = i {
  3. println!("zero");
  4. }

if let 语法格式如下:

  1. if let 匹配值 = 源变量 {
  2. 语句块
  3. }

可以在之后添加一个 else 块来处理例外情况。

if let 语法可以认为是只区分两种情况的 match 语句的"语法糖"(语法糖指的是某种语法的原理相同的便捷替代品)。

对于枚举类依然适用:

  1. fn main() {
  2.     enum Book {
  3.         Papery(u32),
  4.         Electronic(String)
  5.     }
  6.     let book = Book::Electronic(String::from("url"));
  7.     if let Book::Papery(index) = book {
  8.         println!("Papery {}", index);
  9.     } else {
  10.         println!("Not papery book");
  11.     }
  12. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/189238?site
推荐阅读
相关标签
  

闽ICP备14008679号