2014-12-02 08:20:48 -06:00
|
|
|
% Traits
|
|
|
|
|
|
|
|
Do you remember the `impl` keyword, used to call a function with method
|
|
|
|
syntax?
|
|
|
|
|
|
|
|
```{rust}
|
|
|
|
struct Circle {
|
|
|
|
x: f64,
|
|
|
|
y: f64,
|
|
|
|
radius: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Circle {
|
|
|
|
fn area(&self) -> f64 {
|
|
|
|
std::f64::consts::PI * (self.radius * self.radius)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Traits are similar, except that we define a trait with just the method
|
|
|
|
signature, then implement the trait for that struct. Like this:
|
|
|
|
|
|
|
|
```{rust}
|
|
|
|
struct Circle {
|
|
|
|
x: f64,
|
|
|
|
y: f64,
|
|
|
|
radius: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
trait HasArea {
|
|
|
|
fn area(&self) -> f64;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasArea for Circle {
|
|
|
|
fn area(&self) -> f64 {
|
|
|
|
std::f64::consts::PI * (self.radius * self.radius)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
As you can see, the `trait` block looks very similar to the `impl` block,
|
|
|
|
but we don't define a body, just a type signature. When we `impl` a trait,
|
|
|
|
we use `impl Trait for Item`, rather than just `impl Item`.
|
|
|
|
|
|
|
|
So what's the big deal? Remember the error we were getting with our generic
|
|
|
|
`inverse` function?
|
|
|
|
|
|
|
|
```text
|
|
|
|
error: binary operation `==` cannot be applied to type `T`
|
|
|
|
```
|
|
|
|
|
|
|
|
We can use traits to constrain our generics. Consider this function, which
|
|
|
|
does not compile, and gives us a similar error:
|
|
|
|
|
|
|
|
```{rust,ignore}
|
|
|
|
fn print_area<T>(shape: T) {
|
|
|
|
println!("This shape has an area of {}", shape.area());
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Rust complains:
|
|
|
|
|
|
|
|
```text
|
|
|
|
error: type `T` does not implement any method in scope named `area`
|
|
|
|
```
|
|
|
|
|
|
|
|
Because `T` can be any type, we can't be sure that it implements the `area`
|
2015-01-08 18:52:50 -06:00
|
|
|
method. But we can add a *trait constraint* to our generic `T`, ensuring
|
2014-12-02 08:20:48 -06:00
|
|
|
that it does:
|
|
|
|
|
|
|
|
```{rust}
|
|
|
|
# trait HasArea {
|
|
|
|
# fn area(&self) -> f64;
|
|
|
|
# }
|
|
|
|
fn print_area<T: HasArea>(shape: T) {
|
|
|
|
println!("This shape has an area of {}", shape.area());
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
The syntax `<T: HasArea>` means `any type that implements the HasArea trait`.
|
|
|
|
Because traits define function type signatures, we can be sure that any type
|
|
|
|
which implements `HasArea` will have an `.area()` method.
|
|
|
|
|
|
|
|
Here's an extended example of how this works:
|
|
|
|
|
|
|
|
```{rust}
|
|
|
|
trait HasArea {
|
|
|
|
fn area(&self) -> f64;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Circle {
|
|
|
|
x: f64,
|
|
|
|
y: f64,
|
|
|
|
radius: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasArea for Circle {
|
|
|
|
fn area(&self) -> f64 {
|
|
|
|
std::f64::consts::PI * (self.radius * self.radius)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Square {
|
|
|
|
x: f64,
|
|
|
|
y: f64,
|
|
|
|
side: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasArea for Square {
|
|
|
|
fn area(&self) -> f64 {
|
|
|
|
self.side * self.side
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_area<T: HasArea>(shape: T) {
|
|
|
|
println!("This shape has an area of {}", shape.area());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let c = Circle {
|
|
|
|
x: 0.0f64,
|
|
|
|
y: 0.0f64,
|
|
|
|
radius: 1.0f64,
|
|
|
|
};
|
|
|
|
|
|
|
|
let s = Square {
|
|
|
|
x: 0.0f64,
|
|
|
|
y: 0.0f64,
|
|
|
|
side: 1.0f64,
|
|
|
|
};
|
|
|
|
|
|
|
|
print_area(c);
|
|
|
|
print_area(s);
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
This program outputs:
|
|
|
|
|
|
|
|
```text
|
|
|
|
This shape has an area of 3.141593
|
|
|
|
This shape has an area of 1
|
|
|
|
```
|
|
|
|
|
|
|
|
As you can see, `print_area` is now generic, but also ensures that we
|
|
|
|
have passed in the correct types. If we pass in an incorrect type:
|
|
|
|
|
|
|
|
```{rust,ignore}
|
2015-01-13 09:40:18 -06:00
|
|
|
print_area(5);
|
2014-12-02 08:20:48 -06:00
|
|
|
```
|
|
|
|
|
|
|
|
We get a compile-time error:
|
|
|
|
|
|
|
|
```text
|
|
|
|
error: failed to find an implementation of trait main::HasArea for int
|
|
|
|
```
|
|
|
|
|
|
|
|
So far, we've only added trait implementations to structs, but you can
|
|
|
|
implement a trait for any type. So technically, we _could_ implement
|
2015-01-13 09:40:18 -06:00
|
|
|
`HasArea` for `i32`:
|
2014-12-02 08:20:48 -06:00
|
|
|
|
|
|
|
```{rust}
|
|
|
|
trait HasArea {
|
|
|
|
fn area(&self) -> f64;
|
|
|
|
}
|
|
|
|
|
2015-01-13 09:40:18 -06:00
|
|
|
impl HasArea for i32 {
|
2014-12-02 08:20:48 -06:00
|
|
|
fn area(&self) -> f64 {
|
|
|
|
println!("this is silly");
|
|
|
|
|
|
|
|
*self as f64
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-13 09:40:18 -06:00
|
|
|
5.area();
|
2014-12-02 08:20:48 -06:00
|
|
|
```
|
|
|
|
|
|
|
|
It is considered poor style to implement methods on such primitive types, even
|
|
|
|
though it is possible.
|
|
|
|
|
|
|
|
This may seem like the Wild West, but there are two other restrictions around
|
|
|
|
implementing traits that prevent this from getting out of hand. First, traits
|
|
|
|
must be `use`d in any scope where you wish to use the trait's method. So for
|
|
|
|
example, this does not work:
|
|
|
|
|
|
|
|
```{rust,ignore}
|
|
|
|
mod shapes {
|
|
|
|
use std::f64::consts;
|
|
|
|
|
|
|
|
trait HasArea {
|
|
|
|
fn area(&self) -> f64;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Circle {
|
|
|
|
x: f64,
|
|
|
|
y: f64,
|
|
|
|
radius: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasArea for Circle {
|
|
|
|
fn area(&self) -> f64 {
|
|
|
|
consts::PI * (self.radius * self.radius)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let c = shapes::Circle {
|
|
|
|
x: 0.0f64,
|
|
|
|
y: 0.0f64,
|
|
|
|
radius: 1.0f64,
|
|
|
|
};
|
|
|
|
|
|
|
|
println!("{}", c.area());
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Now that we've moved the structs and traits into their own module, we get an
|
|
|
|
error:
|
|
|
|
|
|
|
|
```text
|
|
|
|
error: type `shapes::Circle` does not implement any method in scope named `area`
|
|
|
|
```
|
|
|
|
|
|
|
|
If we add a `use` line right above `main` and make the right things public,
|
|
|
|
everything is fine:
|
|
|
|
|
|
|
|
```{rust}
|
|
|
|
use shapes::HasArea;
|
|
|
|
|
|
|
|
mod shapes {
|
|
|
|
use std::f64::consts;
|
|
|
|
|
|
|
|
pub trait HasArea {
|
|
|
|
fn area(&self) -> f64;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Circle {
|
|
|
|
pub x: f64,
|
|
|
|
pub y: f64,
|
|
|
|
pub radius: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HasArea for Circle {
|
|
|
|
fn area(&self) -> f64 {
|
|
|
|
consts::PI * (self.radius * self.radius)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let c = shapes::Circle {
|
|
|
|
x: 0.0f64,
|
|
|
|
y: 0.0f64,
|
|
|
|
radius: 1.0f64,
|
|
|
|
};
|
|
|
|
|
|
|
|
println!("{}", c.area());
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
This means that even if someone does something bad like add methods to `int`,
|
|
|
|
it won't affect you, unless you `use` that trait.
|
|
|
|
|
|
|
|
There's one more restriction on implementing traits. Either the trait or the
|
|
|
|
type you're writing the `impl` for must be inside your crate. So, we could
|
2015-01-13 09:40:18 -06:00
|
|
|
implement the `HasArea` type for `i32`, because `HasArea` is in our crate. But
|
|
|
|
if we tried to implement `Float`, a trait provided by Rust, for `i32`, we could
|
2014-12-02 08:20:48 -06:00
|
|
|
not, because both the trait and the type aren't in our crate.
|
|
|
|
|
|
|
|
One last thing about traits: generic functions with a trait bound use
|
2015-01-08 18:52:50 -06:00
|
|
|
*monomorphization* (*mono*: one, *morph*: form), so they are statically
|
2015-02-08 19:15:16 -06:00
|
|
|
dispatched. What's that mean? Check out the chapter on [static and dynamic
|
|
|
|
dispatch](static-and-dynamic-dispatch.html) for more.
|
2015-01-13 14:42:38 -06:00
|
|
|
|
2015-02-09 17:00:50 -06:00
|
|
|
## Where clause
|
|
|
|
|
|
|
|
Writing functions with only a few generic types and a small number of trait
|
|
|
|
bounds isn't too bad, but as the number increases, the syntax gets increasingly
|
|
|
|
awkward:
|
|
|
|
|
|
|
|
```
|
|
|
|
use std::fmt::Debug;
|
|
|
|
|
|
|
|
fn foo<T: Clone, K: Clone + Debug>(x: T, y: K) {
|
|
|
|
x.clone();
|
|
|
|
y.clone();
|
|
|
|
println!("{:?}", y);
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
The name of the function is on the far left, and the parameter list is on the
|
|
|
|
far right. The bounds are getting in the way.
|
|
|
|
|
|
|
|
Rust has a solution, and it's called a '`where` clause':
|
|
|
|
|
|
|
|
```
|
|
|
|
use std::fmt::Debug;
|
|
|
|
|
|
|
|
fn foo<T: Clone, K: Clone + Debug>(x: T, y: K) {
|
|
|
|
x.clone();
|
|
|
|
y.clone();
|
|
|
|
println!("{:?}", y);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn bar<T, K>(x: T, y: K) where T: Clone, K: Clone + Debug {
|
|
|
|
x.clone();
|
|
|
|
y.clone();
|
|
|
|
println!("{:?}", y);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
foo("Hello", "world");
|
|
|
|
bar("Hello", "workd");
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
`foo()` uses the syntax we showed earlier, and `bar()` uses a `where` clause.
|
|
|
|
All you need to do is leave off the bounds when defining your type parameters,
|
|
|
|
and then add `where` after the parameter list. For longer lists, whitespace can
|
|
|
|
be added:
|
|
|
|
|
|
|
|
```
|
|
|
|
use std::fmt::Debug;
|
|
|
|
|
|
|
|
fn bar<T, K>(x: T, y: K)
|
|
|
|
where T: Clone,
|
|
|
|
K: Clone + Debug {
|
|
|
|
|
|
|
|
x.clone();
|
|
|
|
y.clone();
|
|
|
|
println!("{:?}", y);
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
This flexibility can add clarity in complex situations.
|
|
|
|
|
|
|
|
`where` is also more powerful than the simpler syntax. For example:
|
|
|
|
|
|
|
|
```
|
|
|
|
trait ConvertTo<Output> {
|
|
|
|
fn convert(&self) -> Output;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ConvertTo<i64> for i32 {
|
2015-02-17 21:00:20 -06:00
|
|
|
fn convert(&self) -> i64 { *self as i64 }
|
2015-02-09 17:00:50 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// can be called with T == i32
|
|
|
|
fn normal<T: ConvertTo<i64>>(x: &T) -> i64 {
|
|
|
|
x.convert()
|
|
|
|
}
|
|
|
|
|
|
|
|
// can be called with T == i64
|
|
|
|
fn inverse<T>() -> T
|
|
|
|
// this is using ConvertTo as if it were "ConvertFrom<i32>"
|
|
|
|
where i32: ConvertTo<T> {
|
|
|
|
1i32.convert()
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
This shows off the additional feature of `where` clauses: they allow bounds
|
|
|
|
where the left-hand side is an arbitrary type (`i32` in this case), not just a
|
|
|
|
plain type parameter (like `T`).
|
|
|
|
|
2015-01-13 14:42:38 -06:00
|
|
|
## Our `inverse` Example
|
|
|
|
|
|
|
|
Back in [Generics](generics.html), we were trying to write code like this:
|
|
|
|
|
|
|
|
```{rust,ignore}
|
|
|
|
fn inverse<T>(x: T) -> Result<T, String> {
|
|
|
|
if x == 0.0 { return Err("x cannot be zero!".to_string()); }
|
|
|
|
|
|
|
|
Ok(1.0 / x)
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
If we try to compile it, we get this error:
|
|
|
|
|
|
|
|
```text
|
|
|
|
error: binary operation `==` cannot be applied to type `T`
|
|
|
|
```
|
|
|
|
|
|
|
|
This is because `T` is too generic: we don't know if a random `T` can be
|
|
|
|
compared. For that, we can use trait bounds. It doesn't quite work, but try
|
|
|
|
this:
|
|
|
|
|
|
|
|
```{rust,ignore}
|
|
|
|
fn inverse<T: PartialEq>(x: T) -> Result<T, String> {
|
|
|
|
if x == 0.0 { return Err("x cannot be zero!".to_string()); }
|
|
|
|
|
|
|
|
Ok(1.0 / x)
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
You should get this error:
|
|
|
|
|
|
|
|
```text
|
|
|
|
error: mismatched types:
|
|
|
|
expected `T`,
|
|
|
|
found `_`
|
|
|
|
(expected type parameter,
|
|
|
|
found floating-point variable)
|
|
|
|
```
|
|
|
|
|
|
|
|
So this won't work. While our `T` is `PartialEq`, we expected to have another `T`,
|
|
|
|
but instead, we found a floating-point variable. We need a different bound. `Float`
|
|
|
|
to the rescue:
|
|
|
|
|
|
|
|
```
|
|
|
|
use std::num::Float;
|
|
|
|
|
|
|
|
fn inverse<T: Float>(x: T) -> Result<T, String> {
|
|
|
|
if x == Float::zero() { return Err("x cannot be zero!".to_string()) }
|
|
|
|
|
|
|
|
let one: T = Float::one();
|
|
|
|
Ok(one / x)
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
We've had to replace our generic `0.0` and `1.0` with the appropriate methods
|
|
|
|
from the `Float` trait. Both `f32` and `f64` implement `Float`, so our function
|
|
|
|
works just fine:
|
|
|
|
|
|
|
|
```
|
|
|
|
# use std::num::Float;
|
|
|
|
# fn inverse<T: Float>(x: T) -> Result<T, String> {
|
|
|
|
# if x == Float::zero() { return Err("x cannot be zero!".to_string()) }
|
|
|
|
# let one: T = Float::one();
|
|
|
|
# Ok(one / x)
|
|
|
|
# }
|
|
|
|
println!("the inverse of {} is {:?}", 2.0f32, inverse(2.0f32));
|
|
|
|
println!("the inverse of {} is {:?}", 2.0f64, inverse(2.0f64));
|
|
|
|
|
|
|
|
println!("the inverse of {} is {:?}", 0.0f32, inverse(0.0f32));
|
|
|
|
println!("the inverse of {} is {:?}", 0.0f64, inverse(0.0f64));
|
|
|
|
```
|