The Step Builder problem
We designed our API. We created a POJO that we want to be immutable. To make it easier to instantiate it, we provide a builder. But to make it even more user friendly, we made it a step builder. So whoever wants to instantiate my POJO can easily concatenate the methods, and the IDE will complete with the correct method time after time.
If you never heard of a step builder, here is an example.
It does not look pretty. But when it is being used it looks just like a regular builder, but the returned interfaces on each step tell the IDE (and the user) which is the next method to call. This also prevents a user from calling the same method multiple times.1
Pretty good. Problem solved.
Wait. I was talking about APIs. What was the point I was trying to make? All this rambling took so long that I can’t even…
Right after so long, that was it. Say after a few months (or the time it took
to get to this point) the Pizzaria decides to make filled crusts. So now you
add a new field to the Pizza
class, it will receive its value from the
Builder
. And the Builder
will add a new step between the crust and the
sauce.
But if the step is mandatory, it will break anyone who had code that did not have crust filling.
Fear not. Since the crust filling relates to the … crust, we could add a new
method to the NeedsCrust
interface.
Ooph, that was a close one. But what if it was something new that does not relate to any of the old fields?
Well, then I guess the way to go would be to add a new method to the Bakeable
interface that returns a Bakeable
. While doing this would keep backwards
compatibility, it allows the user to call that method multiple times. Worst
case scenario you’ll need to add some defensive coding in the Builder
.
We solved the problem. Problem solved.
-
technically the user could always cast the returned type to a
Builder
and access any method. ↩