- How to avoid too much “maybe in the future we’ll expand this…” guessing?
Excerpts from Write Change-Resilient Code With Domain Objects by Google Testing Blog:
This is another post in our Code Health series. A version of this post originally appeared in Google bathrooms worldwide as a Google Testing on the Toilet episode. You can download a printer-friendly version to display in your office.
By Amy Fu
Although a product’s requirements can change often, its fundamental ideas usually change slowly. This leads to an interesting insight: if we write code that matches the fundamental ideas of the product, it will be more likely to survive future product changes.
Domain objects are building blocks (such as classes and interfaces) in our code that match the fundamental ideas of the product. Instead of writing code to match the desired behavior for the product’s requirements (“configure text to be white”), we match the underlying idea (“text color settings”).
For example, imagine you’re part of the gPizza team, which sells tasty, fresh pizzas to feed hungry Googlers. Due to popular demand, your team has decided to add a delivery service.
Without domain objects, the quickest path to pizza delivery is to simply create a deliverPizza method:
public class DeliveryService { public void deliverPizza(List |
---|
Although this works well at first, what happens if gPizza expands its offerings to other foods?
You could add a new method:
public void deliverWithDrinks(List |
---|
But as your list of requirements grows (snacks, sweets, etc.), you’ll be stuck adding more and more methods. How can you change your initial implementation to avoid this continued maintenance burden?
You could add a domain object that models the product’s ideas, instead of its requirements:
- A use case is a specific behavior that helps the product satisfy its business requirements.
(In this case, “Deliver pizzas so we make more money”.) - A domain object represents a common idea that is shared by several similar use cases.
To identify the appropriate domain object, ask yourself:
- What related use cases does the product support, and what do we plan to support in future?
A: gPizza wants to deliver pizzas now, and eventually other products such as drinks and snacks.
- What common idea do these use cases share?
A: gPizza wants to send the customer the food they ordered.
- What is a domain object we can use to represent this common idea?
A: The domain object is a food order. We can encapsulate the use cases in a FoodOrder class.
Domain objects can be a useful generalization - but avoid choosing objects that are too generic, since there is a tradeoff between improved maintainability and more complex, ambiguous code. Generally, aim to support only planned use cases - not all possible use cases (see YAGNI principles).
// GOOD: It’s clear what we’re delivering. public void deliver(FoodOrder order) {} | // BAD: Don’t support furniture delivery. public void deliver(DeliveryList items) {} |
---|
Learn more about domain objects and the more advanced topic of domain-driven design in the book Domain-Driven Design by Eric Evans.