Skip to main content

Value Objects over utility class

· 4 min read

While utility classes offer a way to organize general-purpose functionality, value objects often provide a superior approach, especially for representing domain concepts.

What Are Utility Classes?

Utility classes contain static methods for general-purpose functionality, not tied to specific objects.

While useful, utility classes can lead to:

  • Procedural Code: They encourage a procedural style, separating data and behavior.
  • Anemic Domain Model: They can result in domain objects with little behavior, pushing logic into utility classes.
  • Reduced Readability: Overuse can make code harder to follow as logic is scattered across different classes.

Example: MoneyUtils

Imagine a utility class for handling monetary amounts:

Using Utility classjava
public class MoneyUtils {

  public static BigDecimal add(BigDecimal amount1, Currency currency1, BigDecimal amount2, Currency currency2) {
    if (!currency1.equals(currency2)) {
      throw new IllegalArgumentException("Currencies must match for addition");
    }
    return amount1.add(amount2);
  }

  public static BigDecimal subtract(BigDecimal amount1, Currency currency1, BigDecimal amount2, Currency currency2) {
    if (!currency1.equals(currency2)) {
      throw new IllegalArgumentException("Currencies must match for subtraction");
    }
    return amount1.subtract(amount2);
  }

  public static BigDecimal multiply(BigDecimal amount, Currency currency, BigDecimal multiplier) {
    return amount.multiply(multiplier);
  }

  public static boolean isGreaterThan(BigDecimal amount1, Currency currency1, BigDecimal amount2, Currency currency2) {
    if (!currency1.equals(currency2)) {
      throw new IllegalArgumentException("Currencies must match for comparison");
    }
    return amount1.compareTo(amount2) > 0;
  }
  // potentially other methods for formatting, parsing, etc.
}

This class has several drawbacks:

  • It's not object-oriented.
  • It scatters monetary logic across multiple static methods.
  • It forces you to pass currency codes repeatedly.
  • It doesn't naturally represent the concept of "money" in the domain.

Value Objects: A Better Approach

Value objects are immutable objects that represent a domain concept. They encapsulate data and the behavior associated with that data.

Key characteristics include:

  • Immutability: Their state cannot change after creation.
  • Value Equality: Two value objects are equal if their attributes are equal.
  • Domain Focus: They model concepts from the problem domain.

Example: The Money Value Object

Instead of a static MoneyUtils class, a Money value object encapsulates monetary amounts and currency, unifying data and operations:

Using Value Objectjava
public record Money(BigDecimal amount, Currency currency) {

    public Money add(Money other) {
        if (!this.currency.equals(other.currency)) {
            throw new IllegalArgumentException("Currencies must match for addition");
        }

        return new Money(this.amount.add(other.amount), this.currency);
    }

    public Money subtract(Money other) {
        if (!this.currency.equals(other.currency)) {
            throw new IllegalArgumentException("Currencies must match for subtraction");
        }

        return new Money(this.amount.subtract(other.amount), this.currency);
    }

    public Money multiply(BigDecimal multiplier) {
        return new Money(this.amount.multiply(multiplier), this.currency);
    }

    public boolean isGreaterThan(Money other) {
        if (!this.currency.equals(other.currency)) {
            throw new IllegalArgumentException("Currencies must match for comparison");
        }
        return this.amount.compareTo(other.amount) > 0;
    }

    @Override
    public boolean equals(Object other) {
        if (!(other instanceof Money otherMoney)) {
            return false;
        }
        // Use compareTo to check if amount are numerical equals
        return amount.compareTo(otherMoney.amount) == 0
            && currency.equals(otherMoney.currency);
    }

    @Override
    public int hashCode() {
        // In order to normalize bigdecimal trailing the zeros
        return Objects.hash(amount.stripTrailingZeros(), currency);
    }
}
tip

Always use BigDecimal for monetary operations, and use compareTo

Benefits of using Value Object

  • Domain-Driven Design: Money represents a clear domain concept.
  • Encapsulation: Data and behavior are combined, improving cohesion.
  • Immutability: Ensures that Money objects are thread-safe and predictable.
  • Value Equality: You can easily compare monetary values.
  • Readability: Code becomes more expressive
BigDecimal newAmount = MoneyUtils.add(amount1, currency1, amount2, currency2);

VS

Money money1 = new Money(amount1, currency1);
Money money2 = new Money(amount2, currency2);

Money newAmount = money1.add(money2);

When to Use Value Objects

Prioritize value objects when you are modeling domain concepts, monetary amounts, physical quantities, or any group of data that forms a conceptual whole.

Utility Classes: When They Still Make Sense

Utility classes remain suitable for truly general-purpose, stateless operations that don't belong to a specific domain concept (e.g., pure helper functions).