博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
设计模式 —— 建造者模式
阅读量:6232 次
发布时间:2019-06-22

本文共 4373 字,大约阅读时间需要 14 分钟。

简介

在软件开发中,存在大量复杂对象,它们拥有一系列成员属性,这些成员属性中有些是引用类型的成员对象。而且在这些复杂对象中,还可能存在一些限制条件,如某些属性没有赋值则复杂对象不能作为一个完整的产品使用;有些属性的赋值必须按照某个顺序,一个属性没有赋值之前,另一个属性可能无法赋值等。

由于组合部件的过程很复杂,因此,这些部件的组合过程往往被“外部化”到一个称作建造者的对象里,建造者返还给客户端的是一个已经建造完毕的完整产品对象,而用户无须关心该对象所包含的属性以及它们的组装方式,这就是建造者模式的模式动机。

建造者模式(Builder Pattern)使用多个简单的对象一步一步构建成一个复杂的对象。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

实例

建造者模式包含如下角色:

  1. Builder:抽象建造者
  2. ConcreteBuilder:具体建造者
  3. Director:指挥者
  4. Product:产品

电脑是每个程序员的标配,主要配件有:CPU、主板、内存、显卡、电源。当前主要有两大平台:AMDIntel,但是两个平台的一些零件是不一样的。所以肥宅在组装两个平台时选择的配件也是不一样的。

产品(电脑):

public class Computer {    /**     * cpu     */    private String cpu;    /**     * 主板     */    private String motherboar;    /**     * 内存     */    private String ram;    /**     * 显卡     */    private String graphicsCard;    /**     * 电源     */    private String power;    //setter and getter    // toString}复制代码

抽象工厂:

abstract class AbstractComputerBuilder {    /**     * 选择CPU     */    abstract void buildCPU();    /**     * 选择主板     */    abstract void buildMotherboar();    /**     * 选择内存     */    abstract void buildRam();    /**     * 选择显卡     */    abstract void buildGraphicsCard();    /**     * 选择电源     */    abstract void buildPower();    /**     * 获取电脑     */    abstract Computer getComputer();}复制代码

具体建造者:

// AMD电脑 建造者public class AmdComputerBuilder extends AbstractComputerBuilder {    private Computer computer = new Computer();    @Override    void buildCPU() {        computer.setCpu("2700x");    }    @Override    void buildMotherboar() {        computer.setMotherboar("华硕 ROG X470");    }    @Override    void buildRam() {        computer.setRam("芝奇 DDR4 8G");    }    @Override    void buildGraphicsCard() {        computer.setGraphicsCard("vega 56");    }    @Override    void buildPower() {        computer.setPower("EVGA 750W");    }    @Override    Computer getComputer() {        return computer;    }}// Intel 电脑建造者public class IntelComputerBuilder extends AbstractComputerBuilder {    private Computer computer = new Computer();    @Override    void buildCPU() {        computer.setCpu("i7 8700");    }    @Override    void buildMotherboar() {        computer.setMotherboar("微星 Z370");    }    @Override    void buildRam() {        computer.setRam("金士顿 DDR4 8G");    }    @Override    void buildGraphicsCard() {        computer.setGraphicsCard("GTX 1080Ti");    }    @Override    void buildPower() {        computer.setPower("海韵 750W");    }    @Override    Computer getComputer() {        return computer;    }}复制代码

指挥者:

public class ComputerDirector {    public void construct(AbstractComputerBuilder builder) {        builder.buildCPU();        builder.buildMotherboar();        builder.buildRam();        builder.buildGraphicsCard();        builder.buildPower();    }}复制代码

测试:

public class Fz {    @Test    public void build() {        ComputerDirector director = new ComputerDirector();        AmdComputerBuilder amdComputerBuilder = new AmdComputerBuilder();        director.construct(amdComputerBuilder);        Computer amdComputer = amdComputerBuilder.getComputer();        System.out.println("选择AMD平台配件:" + amdComputer);        IntelComputerBuilder intelComputerBuilder = new IntelComputerBuilder();        director.construct(intelComputerBuilder);        Computer intelComputer = intelComputerBuilder.getComputer();        System.out.println("选择Intel平台配件:" + intelComputer);    }}复制代码

类图

优点

  1. 建造者模式的封装性很好。使用建造者模式可以有效的封装变化,在使用建造者模式的场景中,一般产品类和建造者类是比较稳定的,因此,将主要的业务逻辑封装在导演类中对整体而言可以取得比较好的稳定性。
  2. 在建造者模式中,客户端不必知道产品内部组成的细节,将产品本身与产品的创建过程解耦,使得相同的创建过程可以创建不同的产品对象。
  3. 可以更加精细地控制产品的创建过程。将复杂产品的创建步骤分解在不同的方法中,使得创建过程更加清晰,也更方便使用程序来控制创建过程。
  4. 建造者模式很容易进行扩展。如果有新的需求,通过实现一个新的建造者类就可以完成,基本上不用修改之前已经测试通过的代码,因此也就不会对原有功能引入风险。符合开闭原则。

缺点

  1. 建造者模式所创建的产品一般具有较多的共同点,其组成部分相似,如果产品之间的差异性很大,则不适合使用建造者模式,因此其使用范围受到一定的限制。
  2. 如果产品的内部变化复杂,可能会导致需要定义很多具体建造者类来实现这种变化,导致系统变得很庞大。

适用场景

在以下情况下可以使用建造者模式:

  1. 需要生成的产品对象有复杂的内部结构,这些产品对象通常包含多个成员属性。
  2. 需要生成的产品对象的属性相互依赖,需要指定其生成顺序。
  3. 对象的创建过程独立于创建该对象的类。在建造者模式中引入了指挥者类,将创建过程封装在指挥者类中,而不在建造者类中。
  4. 隔离复杂对象的创建和使用,并使得相同的创建过程可以创建不同的产品。

总结

建造者模式与工厂模式类似,适用的场景也很相似。建造者模式仅仅只比工厂模式多了一个”指挥者”的角色。在建造者模式的类图中,假如把这个导演类看做是最终调用的客户端,那么图中剩余的部分就可以看作是一个简单的工厂模式了。

建造者模式一般用来创建更为复杂的对象,因为对象的创建过程更为复杂,因此将对象的创建过程独立出来组成一个新的类——导演类。也就是说,工厂模式是将对象的全部创建过程封装在工厂类中,由工厂类向客户端提供最终的产品;而建造者模式中,建造者类一般只提供产品类中各个组件的建造,而将具体建造过程交付给导演类。由导演类负责将各个组件按照特定的规则组建为产品,然后将组建好的产品交付给客户端。建造者模式更加关注与零件装配的顺序。

一般来说,如果产品的建造很复杂,那么请用工厂模式;如果产品的建造更复杂,那么请用建造者模式。

转载于:https://juejin.im/post/5c7d47f9e51d453eea5dedd2

你可能感兴趣的文章
作业:实现简单的shell sed替换功能和修改haproxy配置文件
查看>>
Altium 拼板方法以及 注意的 地方
查看>>
Apache Pulsar中的地域复制,第1篇:概念和功能
查看>>
python pip install 出现 OSError: [Errno 1] Operation not permitted
查看>>
从源码分析scrollTo、scrollBy、Scroller方法的区别和作用
查看>>
ObjectOutputStream和ObjectInputStream
查看>>
nagios客户端未启动报错
查看>>
南京大学周志华教授当选欧洲科学院外籍院士
查看>>
马士兵教学语录
查看>>
计算机网络与Internet应用
查看>>
MongodDB学习笔记(二)(复制)
查看>>
oracle在线迁移同步数据,数据库报错
查看>>
linux性能剖析工具
查看>>
PHP模拟发送POST请求之四、加强file_get_contents()发送POST请求
查看>>
搞懂“分布式锁”,看这篇文章就对了
查看>>
给自己出的iOS面试题
查看>>
React as a UI Runtime(四、条件)
查看>>
flutter中的异步
查看>>
IoC容器初始化过程(下)
查看>>
Python GIL
查看>>