一、建造者模式的概念
建造者模式属于创建型设计模式。
指的是将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。
建造者模式主要解决在软件系统中,有时候面临着"一个复杂对象"的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定。
建造者模式主要涉及几个角色:
1、指挥者(Director),负责和客户(Client)对话
2、之后指挥者将客户的产品需求划分为比较稳定的建造过程(AbstractBuilder)
3、指挥者指挥具体的建造者(ConcreteBuilder)干活
4、获取建造者建造的产品给客户
比如组装电脑这个场景,CPU、主板、硬盘和内存等配件是相对稳定的,组装过程也是相当稳定的,先装CPU、内存、硬盘和电源等等(AbstractBuilder),但是配件搭配的方式是多变的(Builder可以多个),如组装家用电脑、游戏电脑。
二、建造者模式的实现
1、先来一个产品类(电脑)
namespace BuilderPattern { using System; using System.Collections.Generic; /// <summary>
/// 电脑类 /// </summary>
public class Computer { /// <summary>
/// 电脑组件集合 /// </summary>
private readonly IList<string> parts = new List<string>(); /// <summary>
/// 把单个组件添加到电脑组件集合中 /// </summary>
/// <param name="part">组件</param>
public void Add(string part) { this.parts.Add(part); } public void Show() { Console.WriteLine("电脑组件清单:"); foreach (var part in this.parts) { Console.WriteLine($"组件:{part}"); } Console.WriteLine("****************"); } } }
2、再来一个抽象Builder、两个具体的Builder
namespace BuilderPattern { /// <summary>
/// 抽象建造者 /// </summary>
public abstract class AbstractBuilder { /// <summary>
/// 装CPU /// </summary>
public abstract void BuildPartCPU(); /// <summary>
/// 装主板 /// </summary>
public abstract void BuildPartMainBoard(); /// <summary>
/// 获得组装好的电脑 /// </summary>
/// <returns>电脑</returns>
public abstract Computer GetComputer(); } }
namespace BuilderPattern { /// <summary>
/// 具体建造者,具体的某个人为具体创建者,例如:装机小王 /// </summary>
public class ConcreteBuilder1 : AbstractBuilder { private readonly Computer computer = new Computer(); public override void BuildPartCPU() { this.computer.Add("CPU1"); } public override void BuildPartMainBoard() { this.computer.Add("Main board1"); } public override Computer GetComputer() { return this.computer; } } }
namespace BuilderPattern { /// <summary>
/// 具体建造者,具体的某个人为具体创建者,例如:装机小李啊 /// </summary>
public class ConcreteBuilder2 : AbstractBuilder { private readonly Computer computer = new Computer(); public override void BuildPartCPU() { this.computer.Add("CPU2"); } public override void BuildPartMainBoard() { this.computer.Add("Main board2"); } public override Computer GetComputer() { return this.computer; } } }
3、指挥者登场(BOSS)
namespace BuilderPattern { /// <summary>
/// 指挥者 /// </summary>
public class Director { /// <summary>
/// 组装电脑 /// </summary>
/// <param name="abstractBuilder">建造者</param>
public void Construct(AbstractBuilder abstractBuilder) { abstractBuilder.BuildPartCPU(); abstractBuilder.BuildPartMainBoard(); } } }
3、使用
namespace BuilderPattern { using System; class Program { static void Main(string[] args) { // 客户找到电脑城老板说要买电脑,这里要装两台电脑 // 创建指挥者和构造者
var director = new Director(); AbstractBuilder b1 = new ConcreteBuilder1(); AbstractBuilder b2 = new ConcreteBuilder2(); Console.WriteLine("开始组装第一台电脑"); director.Construct(b1); Console.WriteLine("获取第一台电脑"); var computer1 = b1.GetComputer(); computer1.Show(); Console.WriteLine("开始组装第二台电脑"); director.Construct(b2); Console.WriteLine("获取第二台电脑"); var computer2 = b2.GetComputer(); computer2.Show(); Console.Read(); } } }
结果如下:
三、总结
以上组装电脑的过程,把电脑零件和电脑组装分离了,电脑组装和电脑组装过程分离了,也就是我们常说的解耦。
BOSS可以随时修改电脑的组装过程,不影响交付给客户的产品;也可以随时换掉组装电脑的人(ConcreteBuilder1和ConcreBuilder2),不再依赖具体的人。也就是不依赖细节,都依赖于抽象,使整个电脑组装过程更稳定。
建造者模式适用于所创建的产品具有较多的共同点,其组成部分相似;如果产品之间的差异性很大,则不适合使用建造者模式,如造飞机和装电脑。