【23种设计模式】建造者模式(四)

设计模式,建造,模式 · 浏览次数 : 146

小编点评

**建造者模式** 建造者模式是一种设计模式,它将一个复杂对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。 **优点:** * 客户端无需知道产品内部组成的细节,简化开发。 * 具体的建造者类之间是相互独立的,容易扩展。 * 由于具体的建造者是独立的,因此可以对建造过程逐步细化,而不对其他的模块产生任何影响。 **缺点:** * 建造过程中可能产生多余的 Build 对象和 Director 类。 * 建造过程可能相对稳定,无法完全应对需求的变化。 **使用建造者模式的场景:** 建造一个汽车、电脑或手机等复杂对象的实例。由于这些产品通常由各种零件组装而成,因此需要定义一个抽象建造者接口,并为不同的产品定义不同的具体建造者实现。 **示例:** ```java // 建造者接口 abstract class Builder { abstract void buildCarDoor(); abstract void buildCarWheel(); abstract void buildCarEngine(); // ...其他方法 ... } // 具体建造者实现 class BuickBuilder implements Builder { private Car buickCar; public BuickBuilder() { this.buickCar = new Car("别克"); } @Override public void buildCarDoor() { // 组装别克车门逻辑 } @Override public void buildCarWheel() { // 组装别克车轮逻辑 } @Override public void buildCarEngine() { // 组装别克车发动机逻辑 } } ``` **总结:** 建造者模式是一个非常有效的设计模式,可以使开发过程更加简洁和高效。但是,在使用建造者模式之前,需要考虑其可能出现的缺点。

正文

前言

在软件系统中,有时候面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定。如何应对这种变化?如何提供一种“封装机制”来隔离出“复杂对象的各个部分”的变化,从而保持系统中的“稳定构建算法”不随着需求改变而改变?那就是今天分享的建造者模式,又叫生成器模式,英文名称是Builder Pattern

建造者模式定义

在现实生活中,我们经常会遇到一些构成比较复杂的物品,比如:汽车、电脑和手机等等。它们是一个复杂的物品,主要是由各种零部件组装而成的,他们的组装过程是固定的。就拿汽车来说,组装流水线是固定的,不变的,需要把底盘、车轮、车门、车灯、发动机引擎、车灯和排气筒等等组装在一起。但是由于需求的变化,这个汽车复杂对象针对不同的品牌,各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定。建造者设计模式定义就是将一个复杂对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。

建造者模式组成

(1)抽象建造者角色(Builder):为创建一个Product对象的各个部件指定抽象接口,以规范产品对象的各个组成成分的建造。一般而言,此角色规定要实现复杂对象的哪些部分的创建,并不涉及具体的对象部件的创建。

(2)具体建造者(ConcreteBuilder)

  • 实现Builder的接口以构造和装配该产品的各个部件。即实现抽象建造者角色Builder的方法。
  • 定义并明确它所创建的表示,即针对不同的商业逻辑,具体化复杂对象的各部分的创建。
  • 提供一个检索产品的接口。
  • 构造一个使用Builder接口的对象即在指导者的调用下创建产品实例。

(3)指导者(Director):调用具体建造者角色以创建产品对象的各个部分。指导者并没有涉及具体产品类的信息,真正拥有具体产品的信息是具体建造者对象。它只负责保证对象各部分完整创建或按某种顺序创建。

(4)产品角色(Product):建造中的复杂对象,对应具体的产品。它要包含那些定义组件的类,包括将这些组件装配成产品的接口。

image

建造者模式代码实现

image

抽象建造者
 /// <summary>
    /// 抽象建造者,它定义了要创建什么部件和最后创建的结果,但是不是组装的的类型
    /// </summary>
    public abstract class Builder
    {
        /// <summary>
        /// 创建车门
        /// </summary>
        public abstract void BuildCarDoor();
        /// <summary>
        /// 创建车轮
        /// </summary>
        public abstract void BuildCarWheel();
        /// <summary>
        /// 创建车引擎
        /// </summary>
        public abstract void BuildCarEngine();


        /*
         * 当然还有部件:
         * 大灯
         * 地盘
         * ....
         * ....
         */

        /// <summary>
        /// 获得组装完成的汽车
        /// </summary>
        /// <returns></returns>
        public abstract Car GetCar();
    }
具体建造者

别克:

 /// <summary>
    /// 具体建造者,具体的车型的建造者,例如:别克
    /// </summary>
    public sealed class BuickBuilder : Builder
    {
        Car buickCar = new Car("别克");
        public override void BuildCarDoor()
        {
            buickCar.Add("Buick's Door");
        }

        public override void BuildCarWheel()
        {
            buickCar.Add("Buick's Wheel");
        }

        public override void BuildCarEngine()
        {
            buickCar.Add("Buick's Engine");
        }

        public override Car GetCar()
        {
            return buickCar;
        }
    }

奥迪:

/// <summary>
    /// 具体建造者,具体的车型的建造者,例如:奥迪
    /// </summary>
    public sealed class AoDiBuilder : Builder
    {
        Car aoDiCar = new Car("奥迪");
        public override void BuildCarDoor()
        {
            aoDiCar.Add("Aodi's Door");
        }

        public override void BuildCarWheel()
        {
            aoDiCar.Add("Aodi's Wheel");
        }

        public override void BuildCarEngine()
        {
            aoDiCar.Add("Aodi's Engine");
        }

        public override Car GetCar()
        {
            return aoDiCar;
        }
    }
汽车固定组装流程定义
 /// <summary>
    /// 汽车类
    /// </summary>
    public sealed class Car
    {
        // 汽车部件集合
        private IList<string> parts = new List<string>();
        //汽车品牌名
        private string _brandName;
        /// <summary>
        /// 构造函数
        /// </summary>
        /// <param name="brandName"></param>
        public Car(string brandName)
        {
            _brandName = brandName;
        }
        /// <summary>
        /// 把单个部件添加到汽车部件集合中
        /// </summary>
        /// <param name="part"></param>
        public void Add(string part)
        {
            parts.Add(part);
        }
        /// <summary>
        /// 汽车组装流程
        /// </summary>
        public void Assembly()
        {
            Console.WriteLine($"{_brandName}汽车开始在组装.......");
            foreach (string part in parts)
            {
                Console.WriteLine("组件" + part + "已装好...");
            }

            Console.WriteLine($"{_brandName}汽车组装完毕.........");
        }
    }
指导者

这里才是调用组装的,Construct方法里面的实现就是创建复杂对象固定算法的实现,根据具体需求变化,控制组装的流程顺序,该算法是固定的,或者说是相对稳定的。

 /// <summary>
    /// 自动化工厂操控台,也就是建造者模式中的指挥者
    /// </summary>
    public class Director
    {
        /// <summary>
        /// 自动化算法指导组装汽车
        /// </summary>
        /// <param name="builder"></param>
        public void Construct(Builder builder)
        {
            builder.BuildCarDoor();
            builder.BuildCarWheel();
            builder.BuildCarEngine();
            /*
             * 指导其他流程
             */
        }
    }
调用
  /// <summary>
        /// 测试方法
        /// </summary>
        public void RunTest()
        {
            Director director = new Director();

            //组装别克
            Builder buickCarBuilder = new BuickBuilder();
            director.Construct(buickCarBuilder);
            Car buickCar = buickCarBuilder.GetCar();
            buickCar.Assembly();

            Console.WriteLine("\r\n*****************************\r\n");

            //组装奥迪
            Builder aoDiCarBuilder = new AoDiBuilder();
            director.Construct(aoDiCarBuilder);
            Car aoDiCar = aoDiCarBuilder.GetCar();
            aoDiCar.Assembly();
        }

image

建造者模式优缺点

优点:

  • 使用建造者模式可以使客户端不必知道产品内部组成的细节

  • 具体的建造者类之间是相互独立的,容易扩展

  • 由于具体的建造者是独立的,因此可以对建造过程逐步细化,而不对其他的模块产生任何影响。

缺点:

  • 产生多余的Build对象以及Director类。

与【23种设计模式】建造者模式(四)相似的内容:

【23种设计模式】建造者模式(四)

## 前言 在软件系统中,有时候面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定。如何应对这种变化?如何提供一种“封装机制”来隔离出“复杂对象的各个部分”的变化,从而保持系统中

【23种设计模式】设计模式综述(开篇)

## 一、设计模式概述: ​ **设计模式(Design pattern)**代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。是一套被反复使用的、多

【23种设计模式】单例模式(一)

## 前言: 单例模式是创建型模式5种中的第1种,**关注对象的创建, 保证一个类仅有一个实例,并且提供一个全局访问点**。在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性、以及良好的效率。如何绕过常规的构造器,提供一种机制来保证一个类只创建一个实例

【23种设计模式】工厂方法模式(二)

## 前言 在讲述之工厂方法模式前,我们来先了解简单工厂模式,简单工厂模式是最简单的设计模式之一,它虽然不属于GoF的23种设计模式,但是应用也较为频繁,同时它也是学习其他创建型模式的基础。下面我们来先了解下简单工厂模式,然后针对它的缺点来引出工厂方法模式。 ## 简单工厂模式定义 **简单工厂模式

【23种设计模式】抽象工厂模式(三)

## 前言 在抽象工厂模式开篇之前,我们先思考一个问题,如果我们要设计一套房子,其他的组件暂时不考虑,我们仅仅考虑房顶、地板、窗户、房门进行设计。什么样的风格暂时未知,可能会有很多种类。可以先设计一套古典风格的房子,再设计一套现代风格的房子,再设计一套欧式风格的房子....这么多套房子需要设计,需求

【23种设计模式】原型模式(五)

## 前言 在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创建这样的类实例,这就会增加创建类的复杂度和创建过程与客户代码复杂的耦合度。如果采用工厂模式来创建这样的实例对象的话,随着产品类的不断增加,导致子类的数量不断增多,也导致了相

【23种设计模式】适配器模式(六)

## 前言 从今天开始我们开始讲【结构型】设计模式,【结构型】设计模式有如下几种:**适配器模式、桥接模式、组合模式、装饰模式、外观模式、享元模式、代理模式**。【创建型】的设计模式解决的是对象创建的问题,那【结构型】设计模式解决的是类和对象的组合关系的问题。 今天我们就开始讲【结构型】设计模式里面

【23种设计模式】桥接模式(七)

## 前言 【**桥接模式**】是【**结构型**】设计模式的第二个模式,也有叫【桥模式】的,英文名称:**Bridge Pattern**。 大家第一次看到这个名称会想到什么呢?我第一次看到这个模式根据名称猜肯定是连接什么东西的。因为桥在我们现实生活中经常是连接着A地和B地,再往后来发展,桥引申为

【23种设计模式】组合模式(八)

前言 组合模式,英文名称是:Composite Pattern。当我们谈到这个模式的时候,有一个物件和这个模式很像,也符合这个模式要表达的意思,那就是“俄罗斯套娃”。“俄罗斯套娃”就是大的瓷器娃娃里面装着一个小的瓷器娃娃,小的瓷器娃娃里面再装着更小的瓷器娃娃,直到最后一个不能再装更小的瓷器娃娃的那个

【23种设计模式】装饰模式(九)

前言 装饰模式,英文名称:Decorator Pattern。我第一次看到这个名称想到的是另外一个词语“装修”,我就说说我对“装修”的理解吧,大家一定要看清楚,是“装修”,不是“装饰”。在房子装修的过程中,各种功能可以相互组合,来增加房子的功用。类似的,如果我们在软件系统中,要给某个类型或者对象增加