# 定义一个枚举

结构体为你提供了一种将相关字段和数据组合在一起的方法,例如具有 width 和 height 的 Rectangle,而枚举为你提供了一种表示某个值是一组可能值之一的方法。例如,我们可能想说 Rectangle 是一组可能形状之一,该集合还包括 Circle 和 Triangle。为此,Rust 允许我们将这些可能性编码为枚举。

让我们看看我们可能想要在代码中表达的情况,看看为什么枚举在这种情况下比结构更有用、更合适。假设我们需要处理 IP 地址。目前,IP 地址使用两个主要标准:v4 和 v6。因为这些是我们的程序会遇到的唯一可能的 IP 地址,所以我们可以枚举所有可能的变量,这就是枚举名称的由来。

任何 IP 地址都可以是 v4 或 v6 地址,但不能同时是两者。IP 地址的这一特性使得枚举数据结构非常合适,因为枚举值只能是其变体之一。v4 和 v6 地址从根本上来说仍然是 IP 地址,因此当代码处理适用于任何类型的 IP 地址的情况时,应将它们视为同一类型。

我们可以通过定义 IpAddrKind 枚举并列出 IP 地址可能的类型 V4 和 V6,在代码中表达这一概念。这些是枚举的变体:

enum IpAddrKind {
    V4,
    V6,
}

IpAddrKind 现在是一种自定义数据类型,我们可以在代码的其他地方使用它。

# 枚举值

我们可以像这样创建 IpAddrKind 的两个变量的实例:

let four = IpAddrKind::V4;
let six = IpAddrKind::V6;

请注意,枚举的变量在其标识符下命名,我们使用双冒号将两者分开。这很有用,因为现在 IpAddrKind::V4IpAddrKind::V6 这两个值都属于同一类型:IpAddrKind。然后,例如,我们可以定义一个接受任何 IpAddrKind 的函数:

fn route(ip_kind: IpAddrKind) {}

我们可以用以下任一方式调用该函数:

route(IpAddrKind::V4);
route(IpAddrKind::V6);

使用枚举还有更多优势。仔细考虑一下我们的 IP 地址类型,目前我们没有办法存储实际的 IP 地址数据;我们只知道它是什么类型。鉴于你刚刚在第 5 章中学习了结构体,你可能会想用结构体来解决这个问题,如示例 6-1 所示。

enum IpAddrKind {
    V4,
    V6,
}
struct IpAddr {
    kind: IpAddrKind,
    address: String,
}
let home = IpAddr {
    kind: IpAddrKind::V4,
    address: String::from("127.0.0.1"),
};
let loopback = IpAddr {
    kind: IpAddrKind::V6,
    address: String::from("::1"),
};

示例 6-1:使用结构体存储 IP 地址的数据和 IpAddrKind 变量

这里,我们定义了一个结构体 IpAddr,它有两个字段:一个类型为 IpAddrKind(我们之前定义的枚举)的 kind 字段和一个类型为 String 的 address 字段。我们有两个此结构体的实例。第一个是 home,它的类型值为 IpAddrKind::V4,关联的地址数据为 127.0.0.1。第二个实例是 loopback。它的类型值为 IpAddrKind 的另一个变量 V6,关联的地址为 ::1。我们使用结构体将类型和地址值捆绑在一起,因此现在变量与值相关联。

但是,仅使用枚举来表示相同的概念更为简洁:我们可以将数据直接放入每个枚举变量中,而不是将枚举放在结构体中。IpAddr 枚举的这个新定义表明 V4 和 V6 变量都将具有关联的字符串值:

enum IpAddr {
    V4(String),
    V6(String),
}
let home = IpAddr::V4(String::from("127.0.0.1"));
let loopback = IpAddr::V6(String::from("::1"));

我们直接将数据附加到枚举的每个变量,因此不需要额外的结构体。在这里,也更容易看到枚举如何工作的另一个细节:我们定义的每个枚举变量的名称也成为构造枚举实例的函数。也就是说,IpAddr::V4() 是一个函数调用,它接受一个 String 参数并返回一个 IpAddr 类型的实例。我们自动获得这个构造函数的定义,这是定义枚举的结果。

使用枚举而不是结构还有另一个优势:每个变量可以具有不同类型和数量的关联数据。版本 4 IP 地址将始终具有四个数字组件,其值介于 0 到 255 之间。如果我们想将 V4 地址存储为四个 u8 值,但仍将 V6 地址表示为一个字符串值,则无法使用结构体。枚举可以轻松处理这种情况:

enum IpAddr {
    V4(u8, u8, u8, u8),
    V6(String),
}
let home = IpAddr::V4(127, 0, 0, 1);
let loopback = IpAddr::V6(String::from("::1"));

我们已经展示了几种定义数据结构来存储版本 4 和版本 6 IP 地址的不同方法。然而,事实证明,想要存储 IP 地址并编码其类型是如此常见,以至于标准库有一个我们可以使用的定义 (opens new window)!让我们看看标准库如何定义 IpAddr:它具有我们定义和使用的确切枚举和变量,但它以两种不同结构的形式将地址数据嵌入到变量中,每个变量的定义不同:

#![allow(unused)]
fn main() {
    struct Ipv4Addr {
        // --snip--
    }

    struct Ipv6Addr {
        // --snip--
    }

    enum IpAddr {
        V4(Ipv4Addr),
        V6(Ipv6Addr),
    }
}

此代码说明,你可以将任何类型的数据放入枚举变量中:例如字符串、数字类型或结构。你甚至可以包含另一个枚举!此外,标准库类型通常不会比你可能想到的复杂得多。

请注意,尽管标准库包含 IpAddr 的定义,但我们仍然可以创建和使用自己的定义而不会发生冲突,因为我们没有将标准库的定义纳入我们的作用域。我们将在第 7 章中进一步讨论如何将类型纳入作用域。

让我们看一下示例 6-2 中的另一个枚举示例:这个示例的变量中嵌入了多种类型。

enum Message {
    Quit,
    Move { x: i32, y: i32 },
    Write(String),
    ChangeColor(i32, i32, i32),
}

示例 6-2:一个 Message 枚举,其每个变量都存储不同数量和类型的值

此枚举有四种不同类型的变体:

  • Quit 根本没有与之关联的数据。
  • Move 有命名字段,就像结构一样。
  • Write 包含单个字符串。
  • ChangeColor 包含三个 i32 值。

定义带有变量的枚举(如示例 6-2 中所示)类似于定义不同类型的结构体定义,只是枚举不使用 struct 关键字,并且所有变体都归入 Message 类型。以下结构体可以保存前面枚举变量保存的相同数据:

struct QuitMessage; // unit struct
struct MoveMessage {
    x: i32,
    y: i32,
}
struct WriteMessage(String); // tuple struct
struct ChangeColorMessage(i32, i32, i32); // tuple struct

但是如果我们使用不同的结构体,每个结构体都有自己的类型,我们就无法像示例 6-2 中定义的 Message 枚举那样轻松地定义一个函数来接收任何这些类型的消息,因为它只有一种类型。

枚举和结构之间还有一个相似之处:就像我们能够使用 impl 在结构上定义方法一样,我们也能够在枚举上定义方法。以下是我们可以在我们的 Message 枚举上定义名为 call 的方法:

impl Message {
    fn call(&self) {
        // method body would be defined here
    }
}
let m = Message::Write(String::from("hello"));
m.call();

方法主体将使用 self 来获取我们调用该方法时的值。在此示例中,我们创建了一个变量 m,其值为 Message::Write(String::from("hello")),当 m.call() 运行时,self 将会位于 call 方法主体中。

让我们看一下标准库中另一个非常常见且有用的枚举:Option。

# Option Enum 及其相对于 Null 值的优势

本节将探讨 Option 的案例研究,它是标准库定义的另一个枚举。Option 类型编码了一种非常常见的情况,即值可以是某物,也可以是空值。

例如,如果你请求非空列表中的第一个项目,你将获得一个值。如果你请求空列表中的第一个项目,你将什么也得不到。用类型系统来表达这个概念意味着编译器可以检查你是否已经处理了所有应该处理的情况;此功能可以防止在其他编程语言中极为常见的错误。

编程语言设计通常考虑要包含哪些功能,但要排除哪些功能也很重要。Rust 没有许多其他语言所具有的 null 功能。Null 是一个表示没有值的值。在具有 null 的语言中,变量始终处于两种状态之一:null 或 not-null。

null 的发明者 Tony Hoare 在 2009 年的演讲“Null 引用:十亿美元的错误”中说道:

我称其为我十亿美元的错误。当时,我正在设计面向对象语言中第一个全面的引用类型系统。我的目标是确保所有引用的使用都绝对安全,并由编译器自动执行检查。但我无法抗拒插入空引用的诱惑,因为它太容易实现了。这导致了无数的错误、漏洞和系统崩溃,在过去四十年中可能造成了十亿美元的损失。

空值的问题在于,如果你尝试将空值用作非空值,你将收到某种错误。由于这种空值或非空属性无处不在,因此极易出现此类错误。

但是,null 试图表达的概念仍然是有用的:null 是由于某种原因当前无效或不存在的值。

问题实际上不在于概念,而在于具体的实现。因此,Rust 没有 null,但它有一个枚举,可以编码值存在或不存在的概念。这个枚举是 Option<T>,它由标准库定义 (opens new window)如下:

enum Option<T> {
    None,
    Some(T),
}

Option<T> 枚举非常有用,甚至包含在 prelude(预导入模块)中;你无需明确将其引入到当前上下文中。它的变体也包含在 prelude 中:你可以直接使用 Some 和 None,而无需使用 Option::前缀。Option<T> 枚举仍然只是一个常规枚举,而 Some(T)None 仍然是 Option<T> 类型的变体。

<T> 语法是我们尚未讨论过的 Rust 特性。它是一个泛型类型参数,我们将在第 10 章中更详细地介绍泛型。现在,你只需要知道 <T> 表示 Option 枚举的 Some 变体可以保存任何类型的一个数据,并且每个代替 T 的具体类型都会使整个 Option 类型成为不同的类型。以下是使用 Option 值保存数字类型和字符串类型的一些示例:

let some_number = Some(5);
let some_char = Some('e');
let absent_number: Option<i32> = None;

some_number 的类型是 Option<i32>。some_char 的类型是 Option<char>,这是另一种类型。Rust 可以推断出这些类型,因为我们在 Some 变量中指定了一个值。对于 Absent_number,Rust 要求我们注明整个 Option 类型:编译器无法仅通过查看 None 值来推断相应的 Some 变量将保留的类型。在这里,我们告诉 Rust,我们的意思是 Absent_number 是 Option<i32> 类型。

当我们有一个 Some 值时,我们知道该值存在并且该值保存在 Some 中。当我们有一个 None 值时,在某种意义上它与 null 含义相同:我们没有有效值。那么为什么 Option<T> 比 null 更好呢?

简而言之,由于 Option<T> 和 T(其中 T 可以是任何类型)是不同的类型,因此编译器不会让我们使用 Option<T> 值,就好像它肯定是一个有效值一样。例如,如下代码不会编译,因为它试图将 i8 添加到 Option<i8>

let x: i8 = 5;
let y: Option<i8> = Some(5);
let sum = x + y;

如果我们运行此代码,我们会收到如下错误消息:

$ cargo run
   Compiling enums v0.1.0 (file:///projects/enums)
error[E0277]: cannot add `Option<i8>` to `i8`
 --> src/main.rs:5:17
  |
5 |     let sum = x + y;
  |                 ^ no implementation for `i8 + Option<i8>`
  |
  = help: the trait `Add<Option<i8>>` is not implemented for `i8`
  = help: the following other types implement trait `Add<Rhs>`:
            `&'a i8` implements `Add<i8>`
            `&i8` implements `Add<&i8>`
            `i8` implements `Add<&i8>`
            `i8` implements `Add`

For more information about this error, try `rustc --explain E0277`.
error: could not compile `enums` (bin "enums") due to 1 previous error

实际上,这个错误信息意味着 Rust 无法理解如何将 i8 和 Option<i8>加在一起,因为它们是不同的类型。当我们在 Rust 中拥有 i8 之类的类型的值时,编译器将确保我们始终拥有有效值。我们可以放心地继续操作,而无需在使用该值之前检查是否为 null。只有当我们拥有 Option<i8>(或我们正在使用的任何类型的值)时,我们才需要担心可能没有值,并且编译器将确保我们在使用该值之前处理这种情况。

换句话说,你必须先将 Option<T> 转换为 T,然后才能对其执行 T 操作。通常,这有助于捕获最常见的 null 问题之一:假设某个值不为 null,而实际上它为 null。

消除错误地假设非空值的风险有助于你对代码更加自信。为了获得可能为空的值,你必须明确选择通过将该值的类型设为 Option<T>。然后,当你使用该值时,你需要明确处理值为空的情况。只要值的类型不是 Option<T>,你就可以安全地假设该值不为空。这是 Rust 经过深思熟虑的设计决定,旨在限制 null 的普遍性并提高 Rust 代码的安全性。

那么,当你拥有 Option<T> 类型的值时,如何从 Some 变体中获取 T 值以便使用该值?Option<T> 枚举具有大量方法,可用于各种情况;你可以在其文档 (opens new window)中查看它们。熟悉 Option<T> 上的方法将对你使用 Rust 的旅程非常有用。

通常,为了使用 Option<T> 值,你需要拥有可以处理每个变体的代码。你需要一些代码,这些代码仅在你拥有 Some(T) 值时运行,并且此代码可以使用内部 T。你希望其他一些代码仅在你拥有 None 值时运行,并且该代码没有可用的 T 值。match 表达式是一个控制流构造,它在与枚举一起使用时就是这样做的:它将根据它拥有的枚举的变体运行不同的代码,并且该代码可以使用匹配值中的数据。