rust/src/doc/trpl/generics.md

125 lines
3.4 KiB
Markdown
Raw Normal View History

% Generics
Sometimes, when writing a function or data type, we may want it to work for
multiple types of arguments. In Rust, we can do this with generics.
Generics are called parametric polymorphism in type theory,
2015-04-18 17:37:49 -04:00
which means that they are types or functions that have multiple forms (poly
is multiple, morph is form) over a given parameter (parametric).
2015-04-18 17:37:49 -04:00
Anyway, enough with type theory, lets check out some generic code. Rusts
standard library provides a type, `Option<T>`, thats generic:
```rust
enum Option<T> {
Some(T),
None,
}
```
2015-04-18 17:37:49 -04:00
The `<T>` part, which youve seen a few times before, indicates that this is
a generic data type. Inside the declaration of our enum, wherever we see a `T`,
2015-04-18 17:37:49 -04:00
we substitute that type for the same type used in the generic. Heres an
example of using `Option<T>`, with some extra type annotations:
2015-04-18 17:37:49 -04:00
```rust
let x: Option<i32> = Some(5);
```
In the type declaration, we say `Option<i32>`. Note how similar this looks to
`Option<T>`. So, in this particular `Option`, `T` has the value of `i32`. On
the right-hand side of the binding, we do make a `Some(T)`, where `T` is `5`.
2015-04-18 17:37:49 -04:00
Since thats an `i32`, the two sides match, and Rust is happy. If they didnt
match, wed get an error:
2015-04-18 17:37:49 -04:00
```rust,ignore
let x: Option<f64> = Some(5);
// error: mismatched types: expected `core::option::Option<f64>`,
// found `core::option::Option<_>` (expected f64 but found integral variable)
```
2015-04-18 17:37:49 -04:00
That doesnt mean we cant make `Option<T>`s that hold an `f64`! They just have
to match up:
2015-04-18 17:37:49 -04:00
```rust
let x: Option<i32> = Some(5);
let y: Option<f64> = Some(5.0f64);
```
This is just fine. One definition, multiple uses.
2015-04-18 17:37:49 -04:00
Generics dont have to only be generic over one type. Consider another type from Rusts standard library thats similar, `Result<T, E>`:
2015-04-18 17:37:49 -04:00
```rust
enum Result<T, E> {
Ok(T),
Err(E),
}
```
This type is generic over _two_ types: `T` and `E`. By the way, the capital letters
2015-04-18 17:37:49 -04:00
can be any letter youd like. We could define `Result<T, E>` as:
2015-04-18 17:37:49 -04:00
```rust
enum Result<A, Z> {
Ok(A),
Err(Z),
}
```
if we wanted to. Convention says that the first generic parameter should be
2015-04-18 17:37:49 -04:00
`T`, for type, and that we use `E` for error. Rust doesnt care, however.
The `Result<T, E>` type is intended to be used to return the result of a
2015-04-18 17:37:49 -04:00
computation, and to have the ability to return an error if it didnt work out.
## Generic functions
We can write functions that take generic types with a similar syntax:
```rust
fn takes_anything<T>(x: T) {
// do something with x
}
```
The syntax has two parts: the `<T>` says “this function is generic over one
type, `T`”, and the `x: T` says “x has the type `T`.”
Multiple arguments can have the same generic type:
```rust
fn takes_two_of_the_same_things<T>(x: T, y: T) {
// ...
}
```
We could write a version that takes multiple types:
```rust
fn takes_two_things<T, U>(x: T, y: U) {
// ...
}
```
Generic functions are most useful with trait bounds, which well cover in the
[section on traits][traits].
[traits]: traits.html
## Generic structs
You can store a generic type in a `struct` as well:
```rust
2015-04-18 17:37:49 -04:00
struct Point<T> {
x: T,
y: T,
}
let int_origin = Point { x: 0, y: 0 };
let float_origin = Point { x: 0.0, y: 0.0 };
```
Similarly to functions, the `<T>` is where we declare the generic parameters,
and we then use `x: T` in the type declaration, too.