Skip to content

Instantly share code, notes, and snippets.

@rohithreddykota
Created January 29, 2025 19:24
Show Gist options
  • Save rohithreddykota/adf19acafd5baa6d2d36650f0fd45c65 to your computer and use it in GitHub Desktop.
Save rohithreddykota/adf19acafd5baa6d2d36650f0fd45c65 to your computer and use it in GitHub Desktop.

Associated Constants in Rust Traits

  1. Trait-Level Constant: A trait can declare one or more constants that implementing types must define.
  2. Per-Implementer Value: Each implementer of the trait can supply a custom value for the constant(s).
  3. Access via T::CONSTANT_NAME: Once implemented, you refer to the constant using the type's namespace (T::CONSTANT_NAME).

Example 1: Multiple Implementers with Different Constant Values

Let’s revisit a slightly elaborated version of your example. Suppose we want each “size” type to declare its own minimum value:

trait MinTrait {
    const MIN_VALUE: i32;
}

// Concrete types
struct Small;
struct Large;

// Implement the trait for each type
impl MinTrait for Small {
    const MIN_VALUE: i32 = 10;
}

impl MinTrait for Large {
    const MIN_VALUE: i32 = 20;
}

// A generic function that uses the trait's associated constant
fn print_min<T: MinTrait>() {
    println!("Min value for this type is: {}", T::MIN_VALUE);
}

fn main() {
    print_min::<Small>(); // prints "Min value for this type is: 10"
    print_min::<Large>(); // prints "Min value for this type is: 20"
}

Explanation

  1. Trait Definition: trait MinTrait { const MIN_VALUE: i32; }
    This trait requires implementers to define a constant named MIN_VALUE.

  2. Implementations:

    • Small sets MIN_VALUE to 10.
    • Large sets MIN_VALUE to 20.
  3. Usage: The function print_min<T: MinTrait> prints T::MIN_VALUE, which is the constant associated with whichever type you substitute for T.


Example 2: Using Associated Constants for Static Configuration

Imagine you have different shapes and need to store the number of sides as a constant:

trait Shape {
    const NUM_SIDES: u32;

    fn describe() {
        println!("This shape has {} sides.", Self::NUM_SIDES);
    }
}

struct Triangle;
struct Square;

impl Shape for Triangle {
    const NUM_SIDES: u32 = 3;
}

impl Shape for Square {
    const NUM_SIDES: u32 = 4;
}

fn main() {
    Triangle::describe(); // prints "This shape has 3 sides."
    Square::describe();   // prints "This shape has 4 sides."
}

Here, each shape must supply its own NUM_SIDES. This helps keep the code well-organized and type-specific, rather than using a global constant or a function that dispatches by checking the shape type.


Trade-Offs and Considerations

  1. Compile-Time Availability: Associated constants are known at compile time. They are not modifiable at runtime.
  2. Use in Generic Code: In generic functions or methods, you can use T::CONSTANT_NAME to retrieve the trait’s constant for any implementer T.
  3. Pattern Matching: You can use associated constants in pattern matching (e.g., matching on array lengths if the constant is used as a size), provided the value is a valid compile-time constant.
  4. Comparison to static/const:
    • A const at module scope is “global” and does not vary by type.
    • Associated constants attach a constant to a specific trait or type, providing a type-specific or trait-specific value.

Further Reading

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment