設計模式之建造者模式——Builder

1、概述

Builder模式,中文名爲建造者模式,又名生成器模式、構建者模式等,是建立型設計模式之一。用於將一個複雜對象的構建與它的表示分離,使得一樣的構建過程能夠建立不一樣的表示。git

1.適用性:

  • 對象的建立比較複雜、有多種建立形式時
  • 建立複雜對象的算法與對象內部組成和裝配是相對獨立的

2.UML類圖

  • Builder:定義建立Product各個部件的抽象接口
  • ConcreteBuilder:繼承自Builder,實現建立具體類型的Product全部部件的接口,並提供一個獲取最終產品的接口。
  • Director:藉助Builder,內部封裝建立產品的具體流程。

3.UML序列圖

  1. Client建立一個ConcreteBuilder
  2. Client經過傳入ConcreteBuilder建立一個Director
  3. Client使用Director構造產品
  4. Client使用ConcreteBuilder組裝並獲取最終產品

2、實例

想象一下,你做爲造物主,你須要創造各類各樣的動物,好比狗和貓。查看源碼

1.在創造前,要先想好狗和貓大致是什麼樣子,很顯然,它們應該有頭、身體和腳。算法

abstract class Animal
{
    public string Head { get; set; }
    public string Body { get; set; }
    public string Foots { get; set; }
    
    public abstract void Display();
}

class Dog : Animal
{
    public override void Display()
    {
        Console.WriteLine("-------------- 狗的基本信息 -------------------");
        Console.WriteLine($"頭:{Head}");
        Console.WriteLine($"身體:{Body}");
        Console.WriteLine($"腳:{Foots}");
    }
}

class Cat : Animal
{
    public override void Display()
    {
        Console.WriteLine("-------------- 貓的基本信息 -------------------");
        Console.WriteLine($"頭:{Head}");
        Console.WriteLine($"身體:{Body}");
        Console.WriteLine($"腳:{Foots}");
    }
}

2.動物的基本特徵肯定了,也就明確了要幹什麼活兒了——建立動物的頭、身體和腳。設計模式

interface IAnimalBuilder
{
    void SetHead();
    void SetBody();
    void SetFoots();
}

3.接下來就是考慮狗和貓各部位的建立細節了。不過,彆着急編碼,先來考慮一件事情:
  DogBuilder裏面除了須要實現IAnimalBuilder的全部接口外,還須要提供一個GetDog()的方法,用來把建立好的Dog給外部;一樣的CatBuilder中也須要提供一個GetCat()方法。這世間動物有多少種?一千?一萬?遠遠不止!在編碼的過程當中很容易就忘記提供這個方法了。因此咱們在抽象Builder和具體Builder中間插入一個抽象層IAnimalBuilder<TAnimal>,全部的ConcreteBuilder都繼承自這個接口。框架

interface IAnimalBuilder<TAnimal> : IAnimalBuilder where TAnimal : Animal 
{
    TAnimal GetAnimal();
}

class DogBuilder : IAnimalBuilder<Dog>
{
    private readonly Dog _dog = new Dog();

    public void SetHead()
    {
        _dog.Head = "狗的頭";
    }

    public void SetBody()
    {
        _dog.Body = "狗的身體";
    }

    public void SetFoots()
    {
        _dog.Foots = "狗的腳";
    }

    public Dog GetAnimal()
    {
        return _dog;
    }
}

class CatBuilder : IAnimalBuilder<Cat>
{
    private readonly Cat _cat = new Cat();

    public void SetHead()
    {
        _cat.Head = "貓的頭";
    }

    public void SetBody()
    {
        _cat.Body = "貓的身體";
    }

    public void SetFoots()
    {
        _cat.Foots = "貓的腳";
    }

    public Cat GetAnimal()
    {
        return _cat;
    }
}

4.最後,只剩Director了,它來規劃Builder要建立哪些東西。ide

class Director
{
    private readonly IAnimalBuilder _builder;

    public Director(IAnimalBuilder builder)
    {
        _builder = builder;
    }

    public void Construct()
    {
        _builder.SetHead();
        _builder.SetBody();
        _builder.SetFoots();
    }
}

大功告成!接下來就嘗試一下吧測試

var dogBuilder = new DogBuilder();
var dogDirector = new Director(dogBuilder);
dogDirector.Construct();
dogBuilder.GetAnimal().Display();

var catBuilder = new CatBuilder();
var catDirector = new Director(catBuilder);
catDirector.Construct();
catBuilder.GetAnimal().Display();

3、變種

實際開發過程當中,常常會遇到一種需求:建立對象時,對象的一些部分能夠自由選擇設置或不設置,可是對象一旦建立完成,便不能對其進行任何修改。例如:ASP.NET Core框架中WebHost的建立。接下來,咱們來模擬一下女媧造人。
1.一樣的,咱們先設定好人的基本屬性ui

class Person
{
    public string Name { get; set; }
    public int Gender { get; set; }
    public int Age { get; set; }
}

2.接下來,咱們來定義IPersonBuilder接口,除了Build方法外,咱們都返回了IPersonBuilder,對於習慣於使用Linq的.Net開發人員來講,再熟悉不過了,它最大的好處就是可以讓咱們使用Fluent的方式來編寫代碼。this

public interface IPersonBuilder
{
    //注意,這裏有問題
    Person Build();
    IPersonBuilder SetName(string name);
    IPersonBuilder SetGender(int gender);
    IPersonBuilder SetAge(int age);
}

你發現什麼問題了嗎?沒錯,這違背了面向對象五大原則之一——依賴倒置(抽象不該該依賴具體);並且,這也不知足「對象建立後不得進行更改」的需求。因此,咱們須要提取出一個抽象層IPersion編碼

public interface IPerson
{
    string Name { get; }
    int Gender { get; }
    int Age { get; }
}

class Person : IPerson
{
    public string Name { get; set; }
    public int Gender { get; set; }
    public int Age { get; set; }
}

public interface IPersonBuilder
{
    IPerson Build();
    IPersonBuilder SetName(string name);
    IPersonBuilder SetGender(int gender);
    IPersonBuilder SetAge(int age);
}

3.下一步就是構建PersonBuilder了,用來實現IPersonBuilder接口的具體邏輯。設計

class PersonBuilder : IPersonBuilder
{
    private readonly Person _person = new Person();

    public IPerson Build()
    {
        return _person;
    }

    public IPersonBuilder SetName(string name)
    {
        _person.Name = name;
        return this;
    }

    public IPersonBuilder SetGender(int gender)
    {
        _person.Gender = gender;
        return this;
    }

    public IPersonBuilder SetAge(int age)
    {
        _person.Age = age;
        return this;
    }      
}

//提供一個輔助類,用來幫 Client 建立 PersonBuilder
public class PersonBuilderHelper
{
    public static IPersonBuilder CreatePersonBuilder()
    {
        return new PersonBuilder();
    }
}

4.Ok,大功告成!來測試一下吧

var person = PersonBuilderHelper.CreatePersonBuilder()
    .SetAge(20)
    .SetName("jjj")
    .Build();
//輸出:jjj,0,20
Console.WriteLine($"{person.Name},{person.Gender},{person.Age}");

4、總結

經典版與變種版本的區別:

  • 經典版傾向於將構建規劃封裝起來,Client不關心內部邏輯;而變種版本消除了Director,使得Client擁有更多的主動權,能夠自由地進行構建。
  • 經典版須要使用Director進行構建,而後使用Builder進行組裝返回結果;變種版本則直接使用Builder進行鏈式構建並組裝返回結果,結構更加清晰明瞭。
  • 經典版經常使用於多種產品的構建是比較固定的狀況,Director種類也不宜過多,且必須適應全部產品;而變種版更傾向於某一種產品的構建,構建方式不固定、至關複雜的狀況。

查看源碼 若是有新發現會進行補充!

相關文章
相關標籤/搜索