0

Builder pattern and Director class

Antonio 1 week ago updated by Alexander 2 days ago 3

Hi, I would like to discuss the role of the Director class in the Builder pattern.

On the website you state:

 Having a director class in your program isn’t strictly necessary. You can always call the building steps in a specific order directly from the client code.

I think that this statement goes against the Builder pattern usefulness for the following reasons:

  1. In the book of the GoF the Director class isn't defined as optional, but as required by the pattern.
  2. How useful would the Builder pattern be without the Director? Internally Builder classes basically call the methods of the product they are building, making them sort of a duplicated code of the product class themselves. 

To elaborate a little more on the second point, given the following classes:

CarManualCarManualBuilder
- seats: number
- doors: number
- color: string
- manual: CarManual
+ setSeats(number)
+ setDoors(doors)
+ setColor(color)
+ setSeats(number)
+ setDoors(doors)
+ setColor(color)
+ getProduct(): CarManual

What are the advantages of doing:

CarManualBuilder builder = new  CarManualBuilder();
builder.setSeats(4);
builder.setDoors(5);
builder.setColor('red');
CarManual manual = builder.getProduct();

Instead of:

CarManual manual = new CarManual();
manual.setSeats(4);
manual.setDoors(5);
manual.setColor('red');

Thank you

Hi!

Let me address the Director question first. 

I'm pretty sure that if you take a group of 100 developers who claim to have used the Builder pattern and ask them to present their production code that uses Builder, you would see a designated Director class a handful of times.

This is because the main convenience of Builder nowadays is to construct products step-by-step with chaining. You would see something like this a lot more often than a full-blown implementation with a proper Director:

CarManual manual = (new  CarManualBuilder())
    .setSeats(4);
    .setDoors(5);
    .setColor('red');
    .getProduct();

In most real-world cases, the products are of the same class hierarchy or interface (otherwise, you what would be the type of getProduct?), so it's fine to have that getProduct on the Builder. The classic implementation deals with the additional problem of having different product types. If you have that problem, you may solve it with the Director. If not, it's just an extra unnecessary class to maintain.

---

The second aspect, "why do you need a builder if you have all the building methods on the product".

The code in the post looks very similar indeed. However, there's an important difference. With the builder, the client only ever has access to the finished product.

If it doesn't feel intuitively that important, imagine that you're creating an API class for some payment provider like Stripe or something. A class like that should be initialized in a certain way: the user (a developer who integrates your API with their own service) has to supply their API keys, there should be some API endpoint set, maybe an API version, etc. When you go a Builder route, you can always validate all the configuration on the getProduct and throw an exception if the user fails to supply something BEFORE letting them call the actual API.

Let me show this with the code. This is the non-Builder version:

ApiClient client = new ApiClientBuilder();
client.key = '...';
// The client instance seems to be ready, but the developer forgot to specify // the API version, for example.

// In some other methods:
client.sendPaymentRequest(...) // The problem with missing API version will only manifest itself on the above call, // with random consequences.

Now, the same thing with Builder

ApiClient client = new ApiClientBuilder()
    .setKey(...)
    .getClient();
// The code can fail here, because in the implementation of getClient, // you can check all the important constraints and fail early if user // missed to call all the necessary build steps. // User never receives an instance of the api client, unless it's valid.

// In some other methods:
client.sendPaymentRequest(...) // This place won't ever be reached.

Hi @Alexander, thank you so much for the thoughtful reply. One more question if I may. In the code example you provided I can't still see the benefits since both would fail at runtime :( Is it just a readability matter?

Yes, they both will fail at runtime, but the reason for failure will be very different. The builder will likely fail at the initialization time of your app, with an exception condition programmed by you (and maybe even handled gracefully). In the other case, the failure will take place when the client is actually used (e.g. someone tries to place an order) with random consequences (maybe the order will be placed, or maybe not; maybe the price will be correct, or maybe not, etc; maybe the problem will manifest itself during refund, or some other time, etc.).