Skip to main content

Builder Pattern

Builder Pattern

Gamma Categorization: Creational Design Patten
Summary: When piecewise object construction is complicated, provide an API for doing it succintly.

Problem examples

App needs to construct multiple type classes
These classes have complicated construction methods
Isolate construction process from application

Solution

Builder Interface or Abstract class
Create Builder classes that implement builder interface for each object type
Director class: uses builder class to create object

Sample Code

Problem: we need to construct a Sales Report, that must be built every day with the sales summary.

Code

So in order to build the report we need an abstract ReportBuilder class, with a simple constructor that initializes the report. The class contains methods for constructing the header, content and footer. The DispatchReport will be implemented for each report type. 

public abstract class ReportBuilder
{
    protected Report report;

    public void CreateReport() => report = new Report();
    public void SetHeader(string header) => report.Header = header;
    public void AppendContentLine(string contentLine) => report.Content.Add(contentLine);
    public void SetFooter(string footer) => report.Footer = footer;

    public abstract string  DispatchReport();

    public void Clear() => report = new Report();
}

The Report object containds a header, multiple content lines and a footer.

public class Report
{
    public string Header { getset; }
    public List<stringContent { getset; } = new List<string>();
    public string Footer { getset; }
}

As an example, we will implement a text report, that will print the report as a string.

public class TextReportBuilder : ReportBuilder
{
    public override string DispatchReport()
    {
        var hr = "-------------------------";

        var sb = new StringBuilder();

        sb.AppendLine(hr);
        sb.AppendLine(report.Header);
        sb.AppendLine(hr);

        foreach (var contentLine in report.Content)
        {
            sb.AppendLine(contentLine);
        }

        sb.AppendLine(hr);
        sb.AppendLine(report.Footer);
        sb.AppendLine(hr);

        return sb.ToString();
    }
}

Usage

var textReportBuilder = new TextReportBuilder();

textReportBuilder.CreateReport();
textReportBuilder.SetHeader($"Daily Report - {DateTime.UtcNow.ToString("dd/MM/yyyy")}");
textReportBuilder.AppendContentLine("14 Hot Meals");
textReportBuilder.AppendContentLine("22 Breakfasts");
textReportBuilder.SetFooter("Copyright - Cafe Shop");

System.Console.WriteLine(textReportBuilder.DispatchReport());

Output
-------------------------
Daily Report - 06/12/2019
-------------------------
14 Hot Meals
22 Breakfasts
-------------------------
Copyright - Cafe Shop
-------------------------


Note: The Builder Patter definition states that you must have a Director object that basicaly encapsulates what we did in the Usage section. It will depend on your context if it makes sense to use a Director class or if you need the flexibility to assemble the report yourself dynamically.

Next, you can proceed to extend the pattern in order chain the build operations using the Fluent Builder Pattern

Comments

Popular posts from this blog

SOLID (4/5) - Interface segregation principle

Interface segregation principle In the field of software engineering, the interface-segregation principle (ISP) states that no client should be forced to depend on methods it does not use. ISP splits interfaces that are very large into smaller and more specific ones so that clients will only have to know about the methods that are of interest to them. Such shrunken interfaces are also called role interfaces. ISP is intended to keep a system decoupled and thus easier to refactor, change, and redeploy. using   System ; namespace   interfacesegregation {      public   class   Document       {     }      public   interface   IMachine       {          void   Print ( Document   d );          void   Scan ( Document   d );     ...

C# Extension Methods

 C# Extension Methods Extension methods      public   static   class   ExtensionMethods     {          public   static   Stopwatch   Measure ( this   Func < int >  f )         {              var   sw  =  new   Stopwatch ();              sw . Start ();              f ();              sw . Stop ();              return   sw ;         }          public   static   void   Save ( this   ISerializable   s...

Configuring Ubuntu

Ubuntu Server Setting up a static IP // https://linuxhint.com/setup_static_ip_address_ubuntu/ // find the network interface name - eg: "enp9s0" ip a sudo nano /etc/netplan/00-installer-config.yaml network: version: 2 ethernets: ens33: addresses: [192.168.1.124/24] gateway4: 192.168.1.254 nameservers: addresses: [1.1.1.1, 8.8.8.8] Connecting to Server ssh root@server_ip_address Create a new user with admin rights adduser username usermod -aG sudo username sudo reboot Disabling Root Login and Limit login attempts(sshd_config) sudo vim /etc/ssh/sshd_config PermitRootLogin no LoginGraceTime 120 # allow only 1 login attempt per connection MaxAuthTries 1 sudo service sshd restart System update sudo apt-get update sudo apt-get upgrade Firewall sudo ufw status sudo ufw allow ssh sudo ufw allow http sudo ufw allow https sudo ufw enable sudo ufw status .NET Core wget https://packages.microsoft.com/config/ubuntu/20.10/packages-microsoft-prod.deb -O packages-...