20.4 Trait Objects: Polymorphism Without Inheritance

Rust’s polymorphism centers on traits. While static dispatch via generics (monomorphization) is often preferred for performance, Rust also supports trait objects for dynamic dispatch, which is conceptually similar to virtual function calls in languages like C++.

20.4.1 Key Features of Trait Objects

  • Dynamic Dispatch: Method calls on a trait object are resolved at runtime through a vtable-like mechanism.
  • Flexible Implementations: Multiple structs can implement the same trait(s) without sharing a base class.
  • Use Cases: Useful when you have an open-ended set of types or need to load implementations dynamically.

20.4.2 Syntax for Trait Objects

Because trait objects may refer to data of unknown size, they must exist behind some form of pointer. Common approaches include:

  • &dyn Trait: A reference to a trait object (borrowed).
  • Box<dyn Trait>: A heap-allocated trait object owned by the Box.

For example:

#![allow(unused)]
fn main() {
trait Animal {
    fn speak(&self);
}
struct Dog;
impl Animal for Dog {
    fn speak(&self) {
        println!("Woof!");
    }
}
fn example(animal: &dyn Animal) {
    animal.speak();
}

let dog = Dog;
example(&dog); // We pass a reference to a type implementing the Animal trait
}

Or:

#![allow(unused)]
fn main() {
trait Animal {
    fn speak(&self);
}
struct Cat;
impl Animal for Cat {
    fn speak(&self) {
        println!("Meow!");
    }
}
let my_animal: Box<dyn Animal> = Box::new(Cat);
my_animal.speak();
}

20.4.3 How Trait Objects Work Internally

A trait object’s “handle” (the part you store in a variable) effectively consists of two pointers:

  1. A pointer to the concrete data (the struct instance).
  2. A pointer to a vtable containing function pointers for the trait’s methods.

When you call a method on a trait object, Rust consults the vtable at runtime to determine the correct function to execute. This grants polymorphism without compile-time awareness of the exact type—at the cost of some runtime overhead.

Example Using Trait Objects

trait Animal {
    fn speak(&self);
}

struct Dog;
struct Cat;

impl Animal for Dog {
    fn speak(&self) {
        println!("Woof!");
    }
}

impl Animal for Cat {
    fn speak(&self) {
        println!("Meow!");
    }
}

fn main() {
    let animals: Vec<Box<dyn Animal>> = vec![
        Box::new(Dog),
        Box::new(Cat),
    ];

    for animal in animals {
        animal.speak(); // Dynamic dispatch via the vtable
    }
}

C++ Comparison:

#include <iostream>
#include <memory>
#include <vector>

class Animal {
public:
    virtual ~Animal() {}
    virtual void speak() const = 0;
};

class Dog : public Animal {
public:
    void speak() const override { std::cout << "Woof!\n"; }
};

class Cat : public Animal {
public:
    void speak() const override { std::cout << "Meow!\n"; }
};

int main() {
    std::vector<std::unique_ptr<Animal>> animals;
    animals.push_back(std::make_unique<Dog>());
    animals.push_back(std::make_unique<Cat>());

    for (const auto& animal : animals) {
        animal->speak();
    }
}

In Rust, each struct implements the Animal trait independently, providing similar polymorphism but bypassing rigid class inheritance.

20.4.4 Object Safety

Not every trait can form a trait object. A trait is object-safe if:

  • It does not require methods using generic parameters in their signatures, and
  • It does not require Self to appear in certain positions (other than as a reference parameter).

These constraints ensure Rust can build a valid vtable for the methods. This concept typically does not arise in class-based OOP, but in Rust it ensures trait objects remain well-defined at runtime.