Domain-Driven Design - Value Objects

In Domain-Driven Design a Value Object is an identityless object that represents a property, for example, a name, address, or a date.
Table of Contents
Characteristics of Value Objects
- Value Objects don’t have an identity, but they are identified by their value.
- Value Objects are immutable.
- Value Objects can have validation.
Value Objects in C#
Value Object base class in C#
public abstract record ValueObject<T> : IEquatable<T> where T : IEquatable<T>
{
protected T Value { get; }
protected ValueObject(T value)
{
this.Validate(value);
this.Value: value;
}
protected abstract void Validate(T value);
public bool Equals(T? other)
{
return this.Value.Equals(other);
}
public static bool operator ==(ValueObject<T> a, T b) => a.Equals(b);
public static bool operator !=(ValueObject<T> a, T b) => !a.Equals(b);
}
About this code snippet:
- It is a
record
: Records (C# 9) are primarily built for better supporting immutable data models. - It is
abstract
: This prevents this record from being instantiated and allows to declare abstract methods. T
extends fromIEquatable
:T
is the basic value type of the Value Object, so ifT
extends fromIEquatable
, it forces that basic value type of the Value Object to be equatable.- It extends from
IEquatable<T>
: Allows comparing this Value Object with its basic data type. - It calls an abstract method called Validate in the constructor: So we are validating the value of the Value Object before creating it.
Value Object example in C#
public sealed record FirstName : ValueObject<string>
{
public const UInt16 MaxLength: 50;
public string Name => this.Value;
private FirstName(string value) : base(value)
{
}
public static FirstName Create(string value)
{
return new FirstName(value);
}
protected override void Validate(string value)
{
if (string.IsNullOrWhiteSpace(value))
throw new ArgumentNullException(nameof(value));
if (value.Length > FirstName.MaxLength)
throw new ArgumentOutOfRangeException(nameof(value));
}
}
About this code snippet:
- It is
sealed
: Prevent another class from extends from this one. - It overrides the Validate method with validation rules.
- No public
constructor
: Other classes cannot create instances of this class, except for nested classes. - To create instances of this Value Object we do not call the constructor directly, but it uses the Static Factory Method pattern instead.
Value Objects in TypeScript
Value Object base class in TypeScript
abstract class ValueObject<T> {
protected readonly value: T;
protected constructor(value: T) {
this.validate(value);
this.value: Object.freeze(value);
}
protected abstract validate(value: T): void;
public equals(other?: ValueObject<T>): boolean {
if (other === null || other === undefined) {
return false;
}
if (other.value === undefined) {
return false;
}
return deepEqual(this, other);
}
}
About this code snippet:
- It is
abstract
: This prevents this class from being instantiated and allows to declare abstract methods. - It calls an abstract method called Validate in the constructor: So we are validating the value of the Value Object before creating it.
- It use
Object.freeze()
so the value can no longer be changed. - It has a
equals
method that usesdeepEqual
, you can read about it in this post.
Value Object example in TypeScript
class FirstName extends ValueObject<string> {
public static readonly MaxLength: 64;
protected validate(value: string): void {
if (!value)
throw new InvalidFirstNameError();
if (value.length > FirstName.MaxLength)
throw new FirstNameIsTooLongError();
}
static create(value: string): FirstName {
return new FirstName(value);
}
public get getFirstName(): string {
return this.value;
}
}
About this code snippet:
- It overrides the Validate method with validation rules.
- No public
constructor
: Other classes cannot create instances of this class. - To create instances of this Value Object we do not call the constructor directly, but it uses the Static Factory Method pattern instead.
Categories
Automation Development tools Infrastructure Kubernetes Programming guide Software architectureTags
Recent Posts
Enable SSH Server in the Windows Subsystem for Linux (WSL)
Restart Kubernetes pods following a schedule using Helm
Restart Kubernetes pods following a schedule using Kubectl
Create an Azure Key Vault with RBAC role assignments using Terraform
Get the download url of the latest GitHub release using Bash or PowerShell