This post advocates for using Ruby's built-in features like Struct
and immutable data structures (via freeze
) to create simple, efficient value objects. It argues against using more complex approaches like dry-struct
or Virtus
for basic cases, highlighting that the lightweight, idiomatic approach often provides sufficient functionality with minimal overhead. The article illustrates how Struct
provides concise syntax for defining attributes and automatic equality and hashing based on those attributes, fulfilling the core requirements of value objects. Finally, it demonstrates how to enforce immutability by freezing instances, ensuring predictable behavior and preventing unintended side effects.
This post advocates for using Ruby's built-in features, specifically Struct
, to create value objects. It argues against using gems like Virtus
or hand-rolling complex classes, emphasizing simplicity and performance. The author demonstrates how Struct
provides concise syntax for defining immutable attributes, automatic equality comparisons based on attribute values, and a convenient way to represent data structures focused on holding values rather than behavior. This approach aligns with Ruby's philosophy of minimizing boilerplate and leveraging existing tools for common patterns. By using Struct
, developers can create lightweight, efficient value objects without sacrificing readability or conciseness.
HN commenters largely criticized the article for misusing or misunderstanding the term "value object." They argued that true value objects are defined by their attributes and compared by value, not identity, using examples like 5 == 5
even if they are different instances of the integer 5
. They pointed out that the author's use of Comparable
and overriding ==
based on specific attributes leaned more towards a Data Transfer Object (DTO) or a record. Some questioned the practical value of the approach presented, suggesting simpler alternatives like using structs or plain Ruby objects with attribute readers. A few commenters offered different ways to implement proper value objects in Ruby, including using the Values
gem and leveraging immutable data structures.
Summary of Comments ( 1 )
https://news.ycombinator.com/item?id=43433648
HN users largely criticized the article for misusing or misunderstanding the term "Value Object." Commenters pointed out that true Value Objects are immutable and compared by value, not identity. They argued that the article's examples, particularly using mutable hashes and relying on
equal?
, were not representative of Value Objects and promoted bad practices. Several users suggested alternative approaches like usingStruct
or creating immutable classes with custom equality methods. The discussion also touched on the performance implications of immutable objects in Ruby and the nuances of defining equality for more complex objects. Some commenters felt the title was misleading, promoting a non-idiomatic approach.The Hacker News post titled "How to create value objects in Ruby – the idiomatic way" (linking to an article about creating value objects in Ruby) has several comments discussing various aspects of value objects, their implementation in Ruby, and alternative approaches.
One commenter points out the inherent tension between true value object semantics (immutable, compared by value) and the performance implications of creating new objects for every modification. They highlight the practical compromise often made in Ruby where objects are treated as if they were value objects, even if they are technically mutable under the hood. This commenter also raises the question of whether the performance cost of true immutability is actually significant in typical Ruby applications.
Another commenter emphasizes the importance of clearly defining equality (
==
) and hash code (hash
) methods when working with value objects in Ruby. They mention that usingStruct
can simplify this process, but caution against overlooking these crucial methods for correct value object behavior.The discussion then delves into specific aspects of Ruby's object model and how it affects value object implementation. One commenter argues against using
dup
for creating modified copies of value objects, preferring explicit constructor calls or factory methods for clarity and control. They also advocate for defining methods that return new instances rather than modifying the existing object in place. Another commenter suggests leveraging thedry-struct
gem, which provides built-in support for immutability and value comparisons. This suggestion sparks a brief comparison ofdry-struct
withValue
andData.define
, two other Ruby gems designed for creating value objects, highlighting the tradeoffs between different approaches.A separate thread within the comments discusses the use of
freeze
for enforcing immutability in Ruby. One commenter cautions against overusingfreeze
, particularly when dealing with nested data structures. They explain thatfreeze
only provides shallow immutability, leaving deeper layers potentially mutable, which can lead to unexpected behavior.Finally, a few comments touch on the broader context of value objects and their relationship to domain-driven design (DDD). They suggest that focusing on the conceptual aspects of value objects, namely their role in representing domain concepts, is more important than the specific implementation details. One commenter highlights the importance of understanding the business logic and how value objects contribute to the overall domain model.