Previously I discussed
Strategy Pattern
Factory Pattern
Abstract Factory Pattern
Observer Pattern
Facade Pattern
Today I'll be discussing the Builder Pattern. I usually don't include UML because they are all over the web and haven't brought that much extra value to my write-ups. But this pattern has a few more parts then the previous patterns I have dicussed, and I feel it will help.
Straight from Wikipedia
The builder pattern is a creational pattern, and the point of it is to encapsulate the complexities of creating a
complicated object. The Builder pattern can also create variations of the complex object.
At first I thought this pattern sounded familiar. It sounded a little bit like the Factory or Factory Method Pattern or maybe Abstract Factory. Its pretty different though and a short definition of each should help illustrate.
Factory - Client calls factory and factory directly returns one of many different types based on the parameters given to it. The clinet doesn't know what it's getting back only that the object inherits from a given abstract class or interface.
Factory Method - Client has an instance of an abscract factory wich defines an abstract method for creating products, but the client instantiates the factory with a conrete instance of that factory. The concrete instance impliments the create method and can create one of many types of the product but only return the parent class or interface that represents the product.
Abstract Factory - There are some similarities to the Factory method pattern here but just remember the Factory method pattern only returns one parent type to the client. In Abstract Facotry the client has a reference to an abstract factory, and instanciates it with a concreate factory that implements the abstract factory. The concrete factory creates concrete products and returns multiple parent types to the client.
Builder Pattern - An abstract builder class defines abstract methods for creating various parts, and a method for returning a product. Concrete builders implement the abstract builder class and are responcible for building the parts, keeping track of their representations and providing a method to return the final product. A director class provides a construction method that calls the part building methods in the concrete builder classes. (In what ever order or with what ever parameters it decides on).
Below is my car demonstration of the Builder pattern. The source code along with all other pattern examples can be found here
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
abstract class AbstractBuilder | |
{ | |
public abstract void BuildChassis(); | |
public abstract void BuildEngine(); | |
public abstract void BuildCab(); | |
public abstract void BuildBody(); | |
public abstract void BuildDriveTrain(); | |
public abstract void BuildSuspension(); | |
public abstract void SetName(); | |
public abstract FordTruck GetVehicle(); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class Director | |
{ | |
public void Construct(AbstractBuilder builder) | |
{ | |
builder.BuildChassis(); | |
builder.BuildBody(); | |
builder.BuildCab(); | |
builder.BuildDriveTrain(); | |
builder.BuildEngine(); | |
builder.BuildSuspension(); | |
builder.SetName(); | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class F250Builder : AbstractBuilder | |
{ | |
FordTruck truck; | |
public F250Builder() | |
{ | |
truck = new FordTruck(); | |
} | |
public override void BuildChassis() | |
{ | |
truck.Chassis = new Chassis(); | |
} | |
public override void BuildEngine() | |
{ | |
truck.Engine = new DieselEngine(); | |
} | |
public override void BuildCab() | |
{ | |
truck.Cab = new Cab(); | |
} | |
public override void BuildBody() | |
{ | |
truck.Body = new Body(); | |
} | |
public override void BuildDriveTrain() | |
{ | |
truck.DriveTrain = new F250DriveTrain(); | |
} | |
public override void BuildSuspension() | |
{ | |
truck.Suspension = new F250Suspension(); | |
} | |
public override void SetName() | |
{ | |
truck.Name = "F250 truck"; | |
} | |
public override FordTruck GetVehicle() | |
{ | |
return truck; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class F350Builder : AbstractBuilder | |
{ | |
FordTruck truck; | |
public F350Builder() | |
{ | |
truck = new FordTruck(); | |
} | |
public override void BuildChassis() | |
{ | |
truck.Chassis = new Chassis(); | |
} | |
public override void BuildEngine() | |
{ | |
truck.Engine = new DieselEngine(); | |
} | |
public override void BuildCab() | |
{ | |
truck.Cab = new Cab(); | |
} | |
public override void BuildBody() | |
{ | |
truck.Body = new Body(); | |
} | |
public override void BuildDriveTrain() | |
{ | |
truck.DriveTrain = new F350DriveTrain(); | |
} | |
public override void BuildSuspension() | |
{ | |
truck.Suspension = new F350Suspension(); | |
} | |
public override void SetName() | |
{ | |
truck.Name = "F350 truck"; | |
} | |
public override FordTruck GetVehicle() | |
{ | |
return truck; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class Program | |
{ | |
static void Main() | |
{ | |
Console.WriteLine("Choose a truck to build: 1. F250, 2. F350"); | |
var line = Console.ReadLine(); | |
var director = new Director(); | |
AbstractBuilder builder; | |
FordTruck truck; | |
if (line == "1") | |
{ | |
builder = new F250Builder(); | |
director.Construct(builder); | |
truck = builder.GetVehicle(); | |
} | |
else | |
{ | |
builder = new F350Builder(); | |
director.Construct(builder); | |
truck = builder.GetVehicle(); | |
} | |
Console.WriteLine(string.Format("You chose an {0}.", truck.Name)); | |
Console.WriteLine(string.Format("Your truck has a {0}, a {1}, a {2}, a {3} {4}, a {5} {6}, and a {7} {8}.", | |
truck.Body.Description, | |
truck.Cab.Description, | |
truck.Chassis.Description, | |
truck.Engine.Name, truck.Engine.Description, | |
truck.DriveTrain.Name, truck.DriveTrain.Description, | |
truck.Suspension.Name, truck.Suspension.Description)); | |
} | |
} |