rust/src/doc/trpl/enums.md

69 lines
2.3 KiB
Markdown
Raw Normal View History

2015-04-07 21:16:02 -05:00
% Enums
2015-05-12 14:34:52 -05:00
An `enum` in Rust is a type that represents data that could be one of
several possible variants:
2015-04-07 21:16:02 -05:00
```rust
2015-05-12 14:34:52 -05:00
enum Message {
Quit,
ChangeColor(i32, i32, i32),
Move { x: i32, y: i32 },
Write(String),
2015-04-07 21:16:02 -05:00
}
```
2015-05-12 14:34:52 -05:00
Each variant can optionally have data associated with it. The syntax for
defining variants resembles the syntaxes used to define structs: you can
have variants with no data (like unit-like structs), variants with named
data, and variants with unnamed data (like tuple structs). Unlike
separate struct definitions, however, an `enum` is a single type. A
value of the enum can match any of the variants. For this reason, an
enum is sometimes called a sum type: the set of possible values of the
enum is the sum of the sets of possible values for each variant.
2015-04-07 21:16:02 -05:00
2015-05-12 14:34:52 -05:00
We use the `::` syntax to use the name of each variant: theyre scoped by the name
of the `enum` itself. This allows both of these to work:
2015-04-07 21:16:02 -05:00
2015-05-12 14:34:52 -05:00
```rust
# enum Message {
# Move { x: i32, y: i32 },
# }
let x: Message = Message::Move { x: 3, y: 4 };
enum BoardGameTurn {
Move { squares: i32 },
Pass,
}
2015-04-07 21:16:02 -05:00
2015-05-12 14:34:52 -05:00
let y: BoardGameTurn = BoardGameTurn::Move { squares: 1 };
2015-04-07 21:16:02 -05:00
```
2015-05-12 14:34:52 -05:00
Both variants are named `Move`, but since theyre scoped to the name of
the enum, they can both be used without conflict.
A value of an enum type contains information about which variant it is,
in addition to any data associated with that variant. This is sometimes
referred to as a tagged union, since the data includes a tag
indicating what type it is. The compiler uses this information to
enforce that youre accessing the data in the enum safely. For instance,
you cant simply try to destructure a value as if it were one of the
possible variants:
2015-04-07 21:16:02 -05:00
```rust,ignore
2015-05-12 14:34:52 -05:00
fn process_color_change(msg: Message) {
let Message::ChangeColor(r, g, b) = msg; // compile-time error
}
2015-04-07 21:16:02 -05:00
```
Both variants are named `Digit`, but since theyre scoped to the `enum` name
there's no ambiguity.
2015-04-07 21:16:02 -05:00
Not supporting these operations may seem rather limiting, but its a limitation
which we can overcome. There are two ways: by implementing equality ourselves,
or by pattern matching variants with [`match`][match] expressions, which youll
learn in the next section. We dont know enough about Rust to implement
equality yet, but well find out in the [`traits`][traits] section.
2015-04-07 21:16:02 -05:00
[match]: match.html
2015-05-12 14:34:52 -05:00
[if-let]: if-let.html