ZetCode

Groovy Traits

last modified March 22, 2025

Traits in Groovy are a dynamic way to share reusable behavior across classes, blending the flexibility of mixins with the structure of interfaces. Unlike Java interfaces, traits can include both method implementations and properties, offering a practical alternative to inheritance for code reuse. This tutorial dives into defining and applying traits with real-world examples.

Defining a Trait

Traits are created with the trait keyword, acting as reusable building blocks. They can hold abstract methods (to enforce implementation), concrete methods (ready-to-use logic), and properties (shared state), making them versatile for modular design.

SimpleTrait.groovy
trait Greetable {
    String name

    String greet() {
        "Hello, ${name ?: 'friend'}!"
    }
}

Here, Greetable defines a name property and a greet method. The method uses Groovy's string interpolation and the Elvis operator to provide a fallback if name is null, making it a practical snippet for user-facing applications like chatbots or profiles.

Using a Trait

Classes adopt traits using the implements keyword, inheriting their properties and methods seamlessly. This allows you to enrich a class with pre-defined behavior without duplicating code.

Person.groovy
class Person implements Greetable {
    Person(String name) {
        this.name = name
    }
}

def person = new Person("Alice")
println person.greet()

The Person class implements Greetable, gaining name and greet. When instantiated with "Alice", calling greet leverages the trait's logic to produce a personalized greeting. This could model a user in a messaging app or a customer in a CRM system.

Multiple Traits

Groovy lets a class implement multiple traits, stacking behaviors like Lego bricks. This is ideal for composing complex objects from modular, reusable pieces without deep inheritance hierarchies.

MultipleTraits.groovy
trait Walkable {
    void walk() {
        println "Walking at a steady pace..."
    }
}

trait Runnable {
    void run() {
        println "Running at full speed!"
    }
}

class Athlete implements Walkable, Runnable {}

def athlete = new Athlete()
athlete.walk()
athlete.run()

Athlete combines Walkable and Runnable, gaining both abilities. This mirrors a sports tracking app where an athlete's movement types are logged distinctly, showcasing how traits modularize behavior for real-world entities.

Overriding Trait Methods

Trait methods can be customized by overriding them in the implementing class, allowing tailored behavior while keeping the trait's default as a fallback or template.

OverrideTraitMethod.groovy
trait Greetable {
    String greet() {
        "Hello from the team!"
    }
}

class Person implements Greetable {
    String name
    Person(String name) { this.name = name }

    @Override
    String greet() {
        "Hi, I'm ${name}, nice to meet you!"
    }
}

def person = new Person("Bob")
println person.greet()

Person overrides Greetable's generic greeting with a personalized one using name. This could represent an employee introducing themselves in a company portal, adapting the trait's baseline behavior to fit a specific need.

Default Methods in Traits

Traits shine by offering default method implementations, reducing boilerplate in classes. These defaults can be used as-is or overridden, providing a practical starting point for common functionality.

DefaultMethod.groovy
trait Loggable {
    void log(String message) {
        println "[${new Date()}] $message"
    }
}

class Service implements Loggable {
    void processOrder(int orderId) {
        log("Processing order #$orderId")
    }
}

def service = new Service()
service.processOrder(123)

Loggable provides a log method with a timestamp, used by Service to track order processing. This mimics logging in an e-commerce system, where the trait's default adds context (time) without extra effort in the class.

Traits with Properties

Traits can define properties, automatically equipping implementing classes with state and accessors (getters/setters), simplifying data management across multiple types.

TraitWithProperty.groovy
trait Named {
    String name
}

class Employee implements Named {
    Employee(String name) {
        this.name = name
    }

    String getDetails() {
        "Employee: $name"
    }
}

def emp = new Employee("Charlie")
println emp.name
println emp.getDetails()

Named contributes a name property to Employee, which builds on it with getDetails. This could model a payroll system where all entities (employees, contractors) share a name field via the trait, ensuring consistency.

Traits and Inheritance

Traits can extend other traits, creating a hierarchy of reusable behavior. This allows you to refine or specialize functionality, stacking enhancements while keeping code DRY (Don't Repeat Yourself).

TraitInheritance.groovy
trait Greetable {
    String greet() {
        "Hello, welcome!"
    }
}

trait PoliteGreetable extends Greetable {
    @Override
    String greet() {
        "Greetings, delighted to meet you!"
    }
}

class Guest implements PoliteGreetable {}

def guest = new Guest()
println guest.greet()

PoliteGreetable extends Greetable, refining the greeting to be more formal. Guest adopts this polished behavior, suitable for a hotel check-in system where courtesy enhances user experience, demonstrating trait layering.

Trait with Abstract Method

Traits can enforce contracts with abstract methods, requiring implementing classes to provide specific logic while supplying reusable defaults elsewhere.

AbstractTrait.groovy
trait Reportable {
    abstract String generateReport()

    String formatReport() {
        "Report: ${generateReport()}"
    }
}

class Sales implements Reportable {
    double total
    Sales(double total) { this.total = total }

    String generateReport() {
        "Sales total: \$$total"
    }
}

def sales = new Sales(1500.75)
println sales.formatReport()

Reportable mandates generateReport but provides formatReport. Sales implements the abstract method, using it in a formatted report. This fits a business dashboard where reports vary by data type but share a consistent presentation.

Trait with State and Logic

Traits can blend properties and methods to manage stateful behavior, offering a complete module reusable across contexts like user authentication.

AuthTrait.groovy
trait Authenticatable {
    boolean isLoggedIn = false

    void login() {
        isLoggedIn = true
        println "User logged in"
    }

    void logout() {
        isLoggedIn = false
        println "User logged out"
    }
}

class Account implements Authenticatable {
    String username
    Account(String username) { this.username = username }
}

def acc = new Account("dave")
acc.login()
println acc.isLoggedIn
acc.logout()
println acc.isLoggedIn

Authenticatable tracks login state and provides login/logout methods. Account uses this for user session management, applicable in a web app where authentication is a shared concern across user types.

Combining Traits with Class Hierarchy

Traits can enhance inherited classes, mixing horizontal reuse with vertical inheritance for powerful, layered designs like in game development.

GameTrait.groovy
trait Jumpable {
    void jump() {
        println "$name jumps high!"
    }
}

class Character {
    String name
    Character(String name) { this.name = name }
}

class Player extends Character implements Jumpable {
    Player(String name) { super(name) }
}

def player = new Player("Mario")
player.jump()

Jumpable adds jumping to Player, which inherits name from Character. This models a game character with both inherited traits (name) and mixin behaviors (jumping), blending OOP paradigms effectively.

Best Practices for Using Traits

Source

Groovy Traits Documentation

In this tutorial, we explored Groovy traits as a robust tool for reusable behavior, blending interface-like contracts with concrete implementations. Through practical examples, we've seen how traits enhance modularity and flexibility in Groovy applications.

Author

My name is Jan Bodnar, and I am a passionate programmer with many years of programming experience. I have been writing programming articles since 2007. So far, I have written over 1400 articles and 8 e-books. I have over eight years of experience in teaching programming.

List all Groovy tutorials.