概述

描述

  • 建造者模式,又称生成器模式。是将一个复杂的对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。

  • 建造者模式将客户端与包含多个组成部分的复杂对象的创建过程分离,客户端无需知道复杂对象的内部组成与装配方式,主需要知道所需的建造者即可。类似工厂方法,但是建造者模式返回一个完整的复杂产品,而抽象工厂模式则返回一系列相关的产品。以汽车为例,工厂方法可以看成不同汽车配件的生成,而建造者模式则可以看成时汽车的组装

套路

  • Builder:抽象建造者

    为创建一个产品对象的各个部件指定抽象接口,一般包含两个接口方法:一个用于创建复杂对象的各个部件;另一个用于返回生成好的复杂对象。它就可以是抽象类,也可以是接口。
  • ConcreteBuilder:具体建造者

    实现了Builder接口,即实现了各个部件的具体构造和装配方法,定义并明确其所创建的复杂对象。
  • Product:产品角色

    被构建的复杂对象,包含多个组成部件。
  • Director:指挥者

    负责安排复杂对象的建造次序,指挥者与抽象建造者之间存在关联关系,可以在其Construct()方法中调用建造者对象的部件构造和装配方法,完成复杂对象的建造。

使用场景

  • 需要生成的产品对象有复杂的内部结构,这些产品对象通常包含多个成员属性。
  • 需要生成的产品对象的属性相互依赖,需要指定其生成顺序。
  • 对象的创建过程独立于创建该对象的类。在建造者模式中引入了指挥者类,将创建过程封装在指挥者类中,而不在建造者类中。
  • 隔离复杂对象的创建和使用,并使得相同的创建过程可以创建不同的产品。
  • 示例
    • 不同门派角色生成,具有不同的服装、武器、发型
    • 不同地图生成,如道路、植被、天气、怪物。
    • 武器组装

优缺点

  • 优点

    • 客户端不必知道产品内部组成的细节,将产品本身与产品的创建过程解耦,使得相同的创建过程可以创建不同的产品对象。
    • 用户使用不同的具体建造者即可得到不同的产品对象 。
    • 可以更加精细地控制产品的创建过程 。将复杂产品的创建步骤分解在不同的方法中,使得创建过程更加清晰,也更方便使用程序来控制创建过程。
    • 增加新的具体建造者无须修改原有类库的代码,指挥者类针对抽象建造者类编程,系统扩展方便,符合“开闭原则”。
  • 缺点
    • 建造者模式所创建的产品一般具有较多的共同点,其组成部分相似。如果产品之间的差异性很大,则不适合使用,因此其使用范围受到一定的限制。
    • 如果产品的内部变化复杂,可能会导致需要定义很多具体建造者类来实现这种变化,导致系统变得很庞大。

UE4 实践

  • 做一个简单得门派生成器

  • 创建具体赋值产品 —— 门派角色角色

    // 复杂产品 —— 门派角色
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API USectPlayer : public UObject
    {
    GENERATED_BODY()
    public:
    // 门派
    FString Sect = TEXT("None");
    // 性别
    FString Render = TEXT("None");
    // 服装
    FString Costume = TEXT("None");
    // 武器
    FString Weapon = TEXT("None"); // 展示信息
    void ShowInfo() {
    UE_LOG(LogTemp, Warning, TEXT(__FUNCTION__" %s %s %s %s"), *Sect, *Render, *Costume, *Weapon);
    }
    };
  • 创建抽象建造者 —— 角色构造器,具体建造者 —— 纯阳角色构造器、七秀角色构造器

    // 抽象建造者 —— 角色构造器
    UCLASS(Abstract)
    class DESIGNPATTERNS_API USectPlayerBuilder : public UObject
    {
    GENERATED_BODY()
    public:
    // 构造函数创建复杂角色实例
    USectPlayerBuilder() { SectPlayer = NewObject<USectPlayer>(); } // 各个构建部分
    virtual void BuilderSect(){}
    virtual void BuilderRender(){}
    virtual void BuilderCostume(){}
    virtual void BuilderWeapon(){} // 返回角色实例
    USectPlayer* GetPlayer() { return SectPlayer; }
    protected:
    USectPlayer* SectPlayer;
    }; // 具体建造者 —— 纯阳角色构造器
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API UChunYangSectPlayerBuilder : public USectPlayerBuilder
    {
    GENERATED_BODY()
    public:
    virtual void BuilderSect() override { SectPlayer->Sect = TEXT("纯阳门派"); }
    virtual void BuilderRender() override { SectPlayer->Render = TEXT("男"); }
    virtual void BuilderCostume() override { SectPlayer->Costume = TEXT("纯阳服装"); }
    virtual void BuilderWeapon() override { SectPlayer->Weapon = TEXT("纯阳武器"); }
    }; // 具体建造者 —— 七秀角色构造器
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API UQiXiuSectPlayerBuilder : public USectPlayerBuilder
    {
    GENERATED_BODY()
    public:
    virtual void BuilderSect() override { SectPlayer->Sect = TEXT("七秀门派"); }
    virtual void BuilderRender() override { SectPlayer->Render = TEXT("女"); }
    virtual void BuilderCostume() override { SectPlayer->Costume = TEXT("七秀服装"); }
    virtual void BuilderWeapon() override { SectPlayer->Weapon = TEXT("七秀武器"); }
    };
  • 创建指挥者(Director) —— 角色创建指挥者

    // 指挥者(Director) —— 角色创建指挥者
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API UPlayerCreateDirectory : public USectPlayerBuilder
    {
    GENERATED_BODY()
    public:
    USectPlayer* Construct(USectPlayerBuilder* SectPlayerBuilder) {
    // 逐步构建复杂角色
    SectPlayerBuilder->BuilderSect();
    SectPlayerBuilder->BuilderRender();
    SectPlayerBuilder->BuilderCostume();
    SectPlayerBuilder->BuilderWeapon(); return SectPlayerBuilder->GetPlayer();
    }
    };
  • 测试调用

    // 调用测试用的Actor
    UCLASS()
    class DESIGNPATTERNS_API APlayerBuilder : public AActor
    {
    GENERATED_BODY()
    public: void BeginPlay() override { // 创建指挥者
    UPlayerCreateDirectory* PlayerCreateDirectory = NewObject<UPlayerCreateDirectory>(); // 指定纯阳角色构造器生成角色
    USectPlayer* ChunYangPlayer = PlayerCreateDirectory->Construct(NewObject<UChunYangSectPlayerBuilder>());
    ChunYangPlayer->ShowInfo(); // 指定七秀角色构造器生成角色
    USectPlayer* QiXiuPlayer = PlayerCreateDirectory->Construct(NewObject<UQiXiuSectPlayerBuilder>());
    QiXiuPlayer->ShowInfo();
    }
    };
  • 调式输出

    LogTemp: Warning: APlayerBuilder::BeginPlay 纯阳门派 男 纯阳服装 纯阳武器
    LogTemp: Warning: APlayerBuilder::BeginPlay 七秀门派 女 七秀服装 七秀武器


参考

【UE4 设计模式】建造者模式 Builder Pattern的更多相关文章

  1. 设计模式 - 建造者模式 Builder Pattern

    简介 场景 在创建复杂对象时,用户无须关心该对象所包含的属性以及它们的组装方式,只需要指定复杂对象的类型和内容就可以构建它们. 模式定义 建造者模式:将一个复杂对象的构建与表示分离,使得同样的构建过程 ...

  2. 23种设计模式--建造者模式-Builder Pattern

    一.建造模式的介绍       建造者模式就是将零件组装成一个整体,用官方一点的话来讲就是将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示.生活中比如说组装电脑,汽车等等这些都是建 ...

  3. 【原】iOS设计模式之:建造者模式Builder Pattern,用于改进初始化参数

    本文主要讨论一下iOS中的Builder Pattern.与网上很多版本不同,本文不去长篇大论地解释建造者模式的概念,那些东西太虚了.设计模式这种东西是为了解决实际问题的,不能为了设计模式而设计模式, ...

  4. 乐在其中设计模式(C#) - 建造者模式(Builder Pattern)

    原文:乐在其中设计模式(C#) - 建造者模式(Builder Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 建造者模式(Builder Pattern) 作者:webabc ...

  5. 【设计模式】建造者模式 Builder Pattern

    前面学习了简单工厂模式,工厂方法模式以及抽象工厂模式,这些都是创建类的对象所使用的一些常用的方法和套路, 那么如果我们创建一个很复杂的对象可上面的三种方法都不太适合,那么“专业的事交给专业人去做”,2 ...

  6. iOS设计模式之:建造者模式Builder Pattern,用于改进初始化参数

    转自:http://www.cnblogs.com/wengzilin/p/4365855.html 本文主要讨论一下iOS中的Builder Pattern.与网上很多版本不同,本文不去长篇大论地解 ...

  7. 设计模式系列之建造者模式(Builder Pattern)——复杂对象的组装与创建

    说明:设计模式系列文章是读刘伟所著<设计模式的艺术之道(软件开发人员内功修炼之道)>一书的阅读笔记.个人感觉这本书讲的不错,有兴趣推荐读一读.详细内容也可以看看此书作者的博客https:/ ...

  8. 二十四种设计模式:建造者模式(Builder Pattern)

    建造者模式(Builder Pattern) 介绍将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 示例用同样的构建过程创建Sql和Xml的Insert()方法和Get()方 ...

  9. 建造者模式(Builder Pattern)

    建造者模式(Builder Pattern) 它可以将多个简单的对象一步一步构建成一个复杂的对象. 意图:将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示. 主要解决:主要解决在软 ...

  10. 设计模式—建造者模式(Builder)

    title: 设计模式-建造者模式 建造者模式(Builder)是一步一步创建一个复杂的对象,它允许用户只通过指定复杂对象的类型和内容就可以构建它们,用户不需要知道内部的具体构建细节.建造者模式属于对 ...

随机推荐

  1. 安装或更新时,pip出错,“No module named ‘pip’”

    解决办法: 在pycharm终端(Terminal)中 首先执行 :python -m ensurepip 然后执行 :python -m pip install --upgrade pip

  2. openwrt开发笔记二:树莓派刷openwrt

    前言及准备 本笔记适用于第一次给树莓派刷openwrt系统的玩家,对刷机过程及注意事项进行了记录,刷机之后对openwrt进行一些简单配置. 使用openwrt源码制作固件需要花费一点时间. 平台环境 ...

  3. Lambda表达式——注重过程的编程思想

    一.使用匿名内部类的匿名对象创建线程和Lambda表达式写法 Lambda表达式写法不用去定义一个Runable接口的实现类: 二.方法入参是一个接口或者接口的实现类 三.对某个类的一些对象实例进行排 ...

  4. linux 下 I/O 多路复用初探

    本文内容整理自B站up主 free-coder 发布的视频:[并发]IO多路复用select/poll/epoll介绍 引入 一般来讲,服务器在处理IO请求(一般指的是socket编程)时,需要对so ...

  5. C语言中的符号重载

    摘自<C专家编程>第二章37页                     C语言中符号的重载 符号 意义 static 在函数内部,表示该变量的值在各个调用间一直保持延续性在函数这一级,表示 ...

  6. UVA 506 System Dependencies(模拟 烂题)

    https://vjudge.net/problem/UVA-506 题目是给出了五种指令,DEPEND.INSTALL.REMOVE.LIST.END,操作的格式及功能如下: DEPEND item ...

  7. jsp页面动态获取系统时间

    最近在做练习时碰到了这样一个问题:"读者选择查询图书相应信息,跳转到书目的详细信息界面,当可借阅数量大于零,点击借阅按钮,提示用户借阅成功,并显示归还日期(三个月),否则提示用户该书可借阅数 ...

  8. PHP中命名空间是怎样的存在?(三)

    这是与命名空间有关的最后一篇.最后还是两个比较简单的内容,是关于命名空间和全局相关的一些类.函数.常量的使用对比.当然,最后我们还会总结一下命名空间的名称解析规则做为这三篇系列文章的结束. 全局空间 ...

  9. mysql给数据库表里某个字段赋随机值

    UPDATE sxz_goods set sales_volume_base = round(rand() * 50) + 1 where sales_volume_base =0 ORDER BY ...

  10. SpringBoot 整合 Elastic Stack 最新版本(7.14.1)分布式日志解决方案,开源微服务全栈项目【有来商城】的日志落地实践

    一. 前言 日志对于一个程序的重要程度不用过多的言语修饰,本篇将以实战的方式讲述开源微服务全栈项目 有来商城 是如何整合当下主流日志解决方案 ELK +Filebeat . 话不多说,先看实现的效果图 ...