Skip to content

Traits

Shared behaviour of objects.

In the example summarize_author does not have a default implementation and needs to be implemented by all objects claiming the trait. summarize has a default implementation and does not need to be implemented by all objects.

Implement traits

pub struct NewsArticle {
  pub author: String,
  pub headline: String,
  pub content: String,
}

// implement the summary trait for the struct NewsArticle
impl Summary for NewsArticle {
  fn summarize_author(&self) -> String {
    format!("{}", self.author)
  }
}

pub struct Tweet {
  pub username: String,
  pub content: String,
  pub reply: bool,
  pub retweet: bool,
}

// implement the summary trait for the struct NewsArticle
impl Summary for Tweet {
  fn summarize_author(&self) -> String {
    format!("@{}", self.username)
  }
  fn summarize(&self) -> String {
    format!("{}, by {}", self.username, self.content)
  }
}

// define the trait "Summary" without the implementation
pub trait Summary {
  fn summarize_author(&self) -> String; // only body without default implementation

  fn summarize(&self) -> String {       // with default implementation
    String::from("(Read more from {} ...)", self.summarize_author())
  }
}

Function needing trait objects

Implement functions needing objects implementing a given trait

// define method for object using a trait with the impl syntax
pub fn notify(item: &impl Summary) {
  println!("Breaking news! {}", item.summarize());
}

// or can be defined like this with the trait-bound syntax
pub fn notify<T: Summary>(item: &T) {
  println!("Breaking news! {}", item.summarize());
}

//---------------------------------------------------------------------------
// needing multiple traits with impl syntax
pub fn notify(item1: &(impl Summary + Display), item2: &impl Summary) {
  // ...
}

// needing multiple traits with the trait-bound syntax
pub fn notify<T: Summary + Display>(item1: &T, item2: &impl T) {
  // ...
}

//---------------------------------------------------------------------------
// Can become quite compilated to read
pub some_function<T: Display + Clone, U: Clone + Debug>(t: &T, u: &U) -> i32 {
 // ...
}

// can be simplified with the where clause
pub some_function<T, U>(t: &T, u: &U) -> i32
 where T: Display + Clone,
       U: Clone + Debug
{
  // ...
}

Return values

fn returns_summarizable() -> impl Summary {  // returns a object implementing the Summary trait
  Tweet {
    username: String::from("horse_ebook"),
    content: String::from("tweet content"),
    reply: false,
    retweet: false,
  }
}

fn main() {
 println!("{}", returns_summarizable().summarize());
}

Conditionally Implement Methods using trait bounds

use std::fmt::Display;

struct Pairt<T> {
  x: T,
  y: T,
}

impl<T> Pair<T> {
  fn new(x: T, y: T) -> Self {
    Self {x,y}
  }
}

impl<T: Display+ PartialOrd> Pair<T> {
  fn cmp_display(&self) {
    if self.x >= self.y {
      println!("The largest member is x= {}", self.x);
    } else {
      println!("The largest member is y= {}", self.y);
    }
  }
}

// implement a trait on a type implementing another trait

// implement ToString trait on types implementing the Display trait
   impl<T: Display> ToString for T {
     //...
   }

Trait implementation

deref trait

struct MyBox<T>(T);

impl<T> MyBox<T> {
  fn new(x: T) -> MyBox<T> {
    MyBox(x)
  }
}

impl<T> DeRef for MyBox<T> {
  type Target = T;

  //fn deref(&self) -> &Self::Target {
  fn deref(&self) -> &T {
    &self.0
  }
}

fn main() {
   let x = 5;
   let y = myBox::new(x); // smart pointer

   assert_eq!(5, x);
   assert_eq!(5, *y);
   // assert_eq!(5, *(y.defref()));  // thats what rust used
}

deref coercion

Deref can happen automatically for types implementing the deref trait. Converts a reference from one type to another type.

Rust does deref coercion in three cases:

  • From &T to &U when T: Deref<Target=U>
  • From &mut T to &mut U when T DerefMut<Target=U>
  • From &mut T to &U when T: Deref<Target=U>
fn main(){
  let m: = MyBox::new(String::from("Rust"));
  hello(&m);
  // &MyBox<String> -> &String -> &str //deref coercion happens in two steps here
}

fn hello(name: &str) {
  println!("Hello, {}!", name)
}

drop trait

Can be implemented at any type and defines what happens if a variable goes out of scope. It is to cleanup allocated memory.

struct CustomSmartPointer {
  data: String,
}

impl Drop for CustomSmartPointer {
  fn drop(&mut self) {
    println:("Dropping CustomSmartPointer with data `{}`!", self.data);
  }
}

fn main() {
  let c = CustomSmartPointer {
    data: String::from("my stuff"),
  }
  let d = CustomSmartPointer {
    data: String::from("other stuff"),
  }

  //optional manually execute drop
  //c.drop() wrong potential twice free'd memory
  drop(c);
  println!("CustomSmartPointers created.");
}

Conversion Traits

Rust provides several traits for type conversions, allowing you to convert between types in a clean, idiomatic way. These include From, Into, TryFrom, TryInto, AsRef, and AsMut. Here's an overview:

From and Into

From

The From trait is used for infallible conversions between types. If you implement From<T> for a type U, you can create a U from a T.

impl From<i32> for String {
    fn from(num: i32) -> Self {
        num.to_string()
    }
}

let num = 42;
let num_string: String = String::from(num); // Converts i32 to String

TryFrom and TryInto

TryFrom

The TryFrom trait is used for fallible conversions, where the conversion might fail and return an error. It returns a `Result.

use std::convert::TryFrom;

impl TryFrom<&str> for i32 {
    type Error = std::num::ParseIntError;

    fn try_from(value: &str) -> Result<Self, Self::Error> {
        value.parse::<i32>()
    }
}

let parsed: Result<i32, _> = i32::try_from("42"); // Ok(42)
let invalid: Result<i32, _> = i32::try_from("abc"); // Err(ParseIntError)

TryInto

The TryInto trait is the reciprocal of TryFrom. If you implement TryFrom<T> for a type U, TryInto<U> is automatically implemented for T.

let parsed: Result<i32, _> = "42".try_into(); // Ok(42)
let invalid: Result<i32, _> = "abc".try_into(); // Err(ParseIntError)

AsRef and AsMut

AsRef

The AsRef trait is used for creating immutable references of one type from another. It’s primarily used for borrowing and works with types like String, Vec<T>, etc.

impl AsRef<str> for String {
    fn as_ref(&self) -> &str {
        self
    }
}

let my_string = String::from("hello");
let my_str: &str = my_string.as_ref(); // Borrow a &str from a String

AsMut

The AsMut trait is similar to AsRef, but it creates mutable references.

impl AsMut<str> for String {
    fn as_mut(&mut self) -> &mut str {
        self
    }
}

let mut my_string = String::from("hello");
let my_str: &mut str = my_string.as_mut(); // Borrow a &mut str from a String

Summary Table

Trait Direction Use Case
From T -> U Infallible conversion from T to U
Into T -> U Reciprocal of From
TryFrom T -> U Fallible conversion from T to U
TryInto T -> U Reciprocal of TryFrom
AsRef T -> &U Borrow an immutable reference to U from T
AsMut T -> &mut U Borrow a mutable reference to U from T