【设计模式深度剖析】【4】【创建型】【建造者模式】| 类比选购汽车的过程,加深理解

news2025/1/16 13:00:24

👈️上一篇:抽象工厂模式    |   下一篇:原型模式👉️

目录

  • 建造者模式
    • 概览
    • 定义
      • 英文原话
      • 直译
      • 如何理解呢?
        • 建造者模式将对象的建造过程给抽象出来了
        • 类比选购汽车
    • 4个角色
      • UML类图
      • 1. 抽象建造者(Builder)角色
      • 2. 具体建造者(Concrete Builder)角色
      • 3. 产品(Product)角色
      • 4. 导演者(Director)角色
      • 代码示例
        • 1. 抽象建造者Builder.java
        • 2. 具体建造者ConcreteBuilder
        • 3. 导演者Director.java
        • 4. 产品类Product.java
        • 5. 测试类:DemoTest.java
    • 优点
    • 使用场景
    • 建造者模式示解析:构造笔记本电脑对象
      • 源码
      • 类图
      • 1. 抽象建造者:ComputerBuilder.java
      • 2. 具体建造者
        • 2.1 具体的某型号电脑建造者: T410Builder.java(建造T410)
        • 2.2 具体的某型号电脑建造者: X201Builder.java(建造X201)
      • 3. 抽象产品类:Computer.java
      • 4. 具体产品类
        • 4.1 具体的某型号电脑产品类:T410.java
        • 4.2 具体的某型号电脑产品类:X201.java
      • 5. 导演者、指挥官、客户端类:ComputerDirector.java
      • 6. 测试类:DemoTest.java

建造者模式

建造者模式(Builder Pattern)也叫做生成器模式。
=>>>>>本文源码<<<<<==

概览

  • 定义

    • 如何理解呢?
      • 类比选购汽车
  • 4个角色

    • 代码示例
  • 优点

  • 使用场景

  • 建造者模式示解析:构造笔记本电脑对象

定义

英文原话

Separate the construction of a complex object from its representation so that the same construction process can create different representations.

直译

将复杂对象的构造与其表示分离,以便相同的构造过程可以创建不同的表示。

如何理解呢?

建造者模式和单例模式、工厂模式一样,也属于创建型设计模式。

创建型模式抽象了实例化的过程,即对象创建的过程。

建造者模式将对象的建造过程给抽象出来了

建造模式的产品类有各种各样、复杂、可配置项,建造者就是抽象了所有可供客户调用的为产品对象进行配置的方法,最后提供一个返回产品实例的方法。

这样客户端(使用该类的就是该类客户端,就是导演者、指挥官角色)可以按需进行配置,调用相应的配置方法即可,最后所有所需要的配置都配置好,执行build()方法进行构造产品对象即可,这样客户就得到了一个自己所需的对象实例。

类比选购汽车

打个比方就是买车(汽车就是一个产品类),

但是裸车是可以选配各种配件的,

比如高级音响,高档皮质座椅、高端轮毂、车漆等等,这些配置你是可选的

选配(选配的过程就是调用建造者提供的各个可选的配置方法)好了

告诉(“告诉”这一动作就是执行build()方法)

4s店(就是建造者,他会为你提车),

最终(你就是客户端,就是导演者、指挥者角色,告诉了建造者你的需求,并让建造者为你提供一个所需的产品实例)

获得了精心选配爱车(该车就是客户按需获得的产品类实例,也就是你获得了一个你需要的产品对象)就会有这些配置

4个角色

UML类图

建造者模式类图

1. 抽象建造者(Builder)角色

接口、或者抽象类,意在制定规范、统一标准(所有的具体建造者都按照我说的来做),具体的实现由实现类或子类(具体建造者)进行实现。

该角色规定了产品建造的各个可选配置(可按需配置),并提供一个构造对象的build()抽象方法

2. 具体建造者(Concrete Builder)角色

对抽象构建者的实现

持有一个产品对象属性,且该类提供多个方法可供外部选择调用,对该产品对象进行配置,然后对外提供一个build()方法,返回根据客户建造的产品对象。

这里和之前工厂模式区分开,简化理解,产品角色不分抽象产品角色和具体产品角色,只有一个产品类。复杂的场景再对产品类进行抽象。

其实本质上就是通过建造者将对象的建造过程给抽象了:建造者类内持有一个原始产品对象,客户(在这里的客户就是导演者、指挥者角色)可以按需配置,最终生产出所需产品对象。

3. 产品(Product)角色

该角色是建造中的复杂对象,有多种多样的、丰富的可配属性。

4. 导演者(Director)角色

即客户端,根据自己需要对要建造的对象进行配置,最后执行build()方法,获得产品实例。

代码示例

=>>>>>代码示例<<<<<==

1. 抽象建造者Builder.java
package com.polaris.designpattern.list1.creational.pattern4.builder.proto;

public abstract class Builder {
    // 设置产品的不同部分,以获得不同的产品
    public abstract void setPart1();
    public abstract void setPart2();
    public abstract void setPart3();
    // ......其他部件

    // 建造产品
    public abstract Product buildProduct();
}
2. 具体建造者ConcreteBuilder
package com.polaris.designpattern.list1.creational.pattern4.builder.proto;

public class ConcreteBuilder extends Builder {

    private Product product = new Product();

    @Override
    public void setPart1() {
        // 为product配置部件1
        System.out.println("set part 1 for the product.");
    }

    @Override
    public void setPart2() {
        // 为product配置部件2
        System.out.println("set part 2 for the product.");
    }

    @Override
    public void setPart3() {
        // 为product配置部件3
        System.out.println("set part 3 for the product.");
    }

    // ......配置其他部件

    // 建造一个产品
    @Override
    public Product buildProduct() {
        return product;
    }
}
3. 导演者Director.java
package com.polaris.designpattern.list1.creational.pattern4.builder.proto;

public class Director {
    private Builder builder = new ConcreteBuilder();
    public Product build(){
        builder.setPart1();
        builder.setPart2();
        builder.setPart3();
        // ......其他部件
        return builder.buildProduct();
    }
}
4. 产品类Product.java
package com.polaris.designpattern.list1.creational.pattern4.builder.proto;

public class Product {
    // 产品类的业务处理方法
    // ...
}
5. 测试类:DemoTest.java
package com.polaris.designpattern.list1.creational.pattern4.builder.proto;

public class DemoTest {
    public static void main(String[] args) {
        Director director = new Director();
        Product product = director.build();
    }
}

/* Output:
set part 1 for the product.
set part 2 for the product.
set part 3 for the product.
*///:~

优点

建造者模式的优点有以下几个方面。

  1. 封装性,使用建造者模式可以使客户端不必知道产品内部组成的细节。
  2. 建造者独立,容易扩展.
  3. 便于控制细节风险,由于具体的建造者是独立的,因此可以对建造过程逐步细化,而不对其他的模块产生任何影响。

使用场景

使用建造者模式的典型场景如下。

  1. 相同的方法,不同的执行顺序,产生不同的结果时,可以采用建造者模式。
  2. 多个部件或零件,都可以装配到一个对象中,但是产生的运行结果又不相同时,则可以使用该模式。
  3. 产品类非常复杂,或者产品类中的方法调用顺序不同产生了不同的效果,这个时候使用建造者模式。
  4. 在对象创建过程中会使用到系统的一些其他对象,这些对象在产品对象的创建过程中不易得到时,也可以采用建造者模式封装该对象的创建过程。该种场景只能是一个补偿方法,因为一个对象不容易获得,而在设计阶段没有发现,要通过创建者模式柔化创建过程,本身已经违反设计的最初目标。
  • [注]:建造者模式关注的是零件类型和装配工艺顺序,这是与工厂方法模式最大的不同之处,虽然同为创建类模式,但是重点不同。

建造者模式示解析:构造笔记本电脑对象

源码

=>>>>>代码示例<<<<<==

类图

请添加图片描述

  • 产品类可以对产品进行抽象(抽象类),抽象(定义)共有属性,具体产品特有属性在具体产品中在声明
  • 具体产品继承抽象产品类,某具体产品独有的属性在具体产品类中声明
  • 抽象建设者不仅提供设置产品实例公共属性的方法,还要声明个别产品自己独有的属性设置方法
  • 具体建设者可以不止一个(用于构建不同的产品),但都要实现抽象构建者的接口,每个具体产品对应一个具体建设者。

1. 抽象建造者:ComputerBuilder.java

注意有的笔记本有独立显卡,有的笔记本没有独立显卡,因此配置独显是个别产品独有的

但是,抽象建造者也需要声明(规定)该属性的配置方法。

package com.polaris.designpattern.list1.creational.pattern4.builder;

public interface ComputerBuilder {
    void buildCpu();//配置CPU

    void buildRam();//配置内存

    void buildHardDisk();//配置硬盘

    void buildGraphicCard();//***|配置显卡|****

    void buildMonitor();//配置显示器

    void buildOs();//配置操作系统

    Computer build();//得到配置好的计算机
}

2. 具体建造者

2.1 具体的某型号电脑建造者: T410Builder.java(建造T410)

T410有配备独显

package com.polaris.designpattern.list1.creational.pattern4.builder;


public class T410Builder implements ComputerBuilder {

    private T410 computer = new T410();

    @Override
    public void buildCpu() {
        computer.setCpu("i5-450");
    }

    @Override
    public void buildRam() {
        computer.setRam("4G 1333MHz");
    }

    @Override
    public void buildHardDisk() {
        computer.setHardDisk("500GB 7200转");
    }

    @Override
    public void buildGraphicCard() {
        computer.setGraphicCard("Nvidia NVS 3100M");//***配置独显***
    }

    @Override
    public void buildMonitor() {
        computer.setMonitor("14英寸 1280*800");
    }

    @Override
    public void buildOs() {
        computer.setOs("Windows 7 旗舰版");
    }

    @Override
    public T410 build() {
        return computer;
    }
}

2.2 具体的某型号电脑建造者: X201Builder.java(建造X201)

X201没有独显

package com.polaris.designpattern.list1.creational.pattern4.builder;


public class X201Builder implements ComputerBuilder {

    private X201 computer = new X201();

    @Override
    public void buildCpu() {
        computer.setCpu("i3-350");
    }

    @Override
    public void buildRam() {
        computer.setRam("2G 1333MHz");
    }

    @Override
    public void buildHardDisk() {
        computer.setHardDisk("250GB 5400转");
    }

    @Override
    public void buildGraphicCard() {
        // 无显卡
        // X201没有独立显卡,没有独显属性,不用操作
    }

    @Override
    public void buildMonitor() {
        computer.setMonitor("12英寸 1280*800");
    }

    @Override
    public void buildOs() {
        computer.setOs("Windows 7 Home版");
    }

    @Override
    public X201 build() {
        return computer;
    }
}

3. 抽象产品类:Computer.java

对电脑配置进行抽象

package com.polaris.designpattern.list1.creational.pattern4.builder;

import lombok.Data;


@Data
public abstract class Computer {
    private String type;
    private String cpu;
    private String ram;
    private String hardDisk;
    private String monitor;
    private String os;
}

4. 具体产品类

4.1 具体的某型号电脑产品类:T410.java

该机型配备独显属性

继承了抽象产品类的所有属性,并声明了特有的属性-显卡

package com.polaris.designpattern.list1.creational.pattern4.builder;

import lombok.Getter;
import lombok.Setter;


public class T410 extends Computer {
    @Getter
    @Setter
    private String graphicCard; //***有独显***

    public T410() {
        this.setType("ThinkPad T410i");
    }

    @Override
    public String toString() {
        return "型号:\t" + this.getType() + "\nCPU:\t" + this.getCpu()
                + "\n内存:\t" + this.getRam() + "\n硬盘:\t" + this.getHardDisk()
                + "\n显卡:\t" + this.getGraphicCard() + "\n显示器:\t" + this.getMonitor()
                + "\n操作系统:\t" + this.getOs();
    }
}
4.2 具体的某型号电脑产品类:X201.java

该机型没有配备独显属性

继承了抽象产品类的所有属性,并且没有自己特有的属性

package com.polaris.designpattern.list1.creational.pattern4.builder;

public class X201 extends Computer {
    public X201() {
        this.setType("ThinkPad X201i");
    }

    @Override
    public String toString() {
        return "型号:\t" + this.getType() + "\nCPU:\t" + this.getCpu()
                + "\n内存:\t" + this.getRam() + "\n硬盘:\t" + this.getHardDisk()
                + "\n显示器:\t" + this.getMonitor()
                + "\n操作系统:\t" + this.getOs();
    }
}

5. 导演者、指挥官、客户端类:ComputerDirector.java

之所以提供多个名字,意在方便理解。其实就是建造者的客户端,用来告诉建造者如何构造对象,并进行构建。

package com.polaris.designpattern.list1.creational.pattern4.builder;

public class ComputerDirector {
    ComputerBuilder builder;

    public T410 constructT410() {
        builder = new T410Builder();
        builder.buildCpu();
        builder.buildRam();
        builder.buildHardDisk();
        builder.buildGraphicCard();
        builder.buildMonitor();
        builder.buildOs();
        return (T410) builder.build();
    }

    public X201 constructX210() {
        builder = new X201Builder();
        builder.buildCpu();
        builder.buildRam();
        builder.buildHardDisk();
        builder.buildMonitor();
        builder.buildOs();
        return (X201) builder.build();
    }
}

6. 测试类:DemoTest.java

package com.polaris.designpattern.list1.creational.pattern4.builder;

public class DemoTest {
    public static void main(String[] args) {
        ComputerDirector director = new ComputerDirector();
        Computer t410 = director.constructT410();
        System.out.println(t410);
        System.out.println("--------------------------");
        Computer x210 = director.constructX210();
        System.out.println(x210);
    }
}

/* Output:

型号:	ThinkPad T410i
CPU:	i5-450
内存:	4G 1333MHz
硬盘:	500GB 7200转
显卡:	Nvidia NVS 3100M
显示器:	14英寸 1280*800
操作系统:	Windows 7 旗舰版
--------------------------
型号:	ThinkPad X201i
CPU:	i3-350
内存:	2G 1333MHz
硬盘:	250GB 5400转
显示器:	12英寸 1280*800
操作系统:	Windows 7 Home版

*///~

👈️上一篇:抽象工厂模式    |   下一篇:原型模式👉️

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1690955.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

02.爬虫---HTTP基本原理

02.HTTP基本原理 1.URI 和 URL 的区别2.HTTP 和 HTTPS 的区别3.请求过程 1.URI 和 URL 的区别 URL&#xff08;Uniform Resource Locator&#xff09;即-统一资源定位符 URL是用来定位和访问互联网上资源的独特标识&#xff0c;它包括了资源的位置&#xff08;如IP地址或域名&a…

Docker-制作镜像的两种方式=》基于容器制作基于Dockerfile制作

本文主要是基于Docker如何制作一个Java镜像&#xff0c;而一个Java镜像想要运行需要系统环境&#xff0c;JDK。所以我们要先有一个系统环境&#xff0c;本文使用的是centos7&#xff0c;JDK选择版本是8&#xff0c;而我使用的Java程序是一个简易的springBoot项目&#xff0c;你…

详细分析Vue按钮间距的优化方式(附Demo)

目录 前言1. 按钮间距2. 垂直间距 前言 正常间距如下&#xff1a; 其Demo如下&#xff1a; <el-table-column label"操作" align"center"><template #default"scope"><el-buttonlinktype"primary"click"openFor…

Node.js —— 前后端的身份认证 之用 express 实现 JWT 身份认证

JWT的认识 什么是 JWT JWT&#xff08;英文全称&#xff1a;JSON Web Token&#xff09;是目前最流行的跨域认证解决方案。 JWT 的工作原理 总结&#xff1a;用户的信息通过 Token 字符串的形式&#xff0c;保存在客户端浏览器中。服务器通过还原 Token 字符串的形式来认证用…

Mysql注入详细讲解

特殊字符 0x3a:0x7e~0x23# 注入基础 联合查询注入(union) :::tips 页面将SQL查询内容显示出来&#xff0c;即为有回显&#xff0c;可以尝试联合查询注入 利用关键字union &#xff0c;union all 拼接恶意SQL语句 ::: 注入流程 有报错&#xff0c;可以利用报错。如&#xff…

[datawhale202405]从零手搓大模型实战:TinyAgent

结论速递 TinyAgent项目实现了一个简单的Agent智能体&#xff0c;主要是实现了ReAct策略&#xff08;推理调用工具的能力&#xff09;&#xff0c;及封装了一个Tool。 项目实现有一定的疏漏。为了正确运行代码&#xff0c;本次对代码Agent部分进行了简单修改&#xff08;完善…

VBA语言専攻每周通知20240524

通知20240524 各位学员∶本周MF系列VBA技术资料增加611-615讲&#xff0c;T3学员看到通知后请免费领取,领取时间5月24日晚上18:00-5月26日晚上18:00。本次增加内容&#xff1a; MF611:用InputBox录入日期 MF612:信息提示10秒后关自动关闭 MF613:只是信息提示10秒 MF614:显…

Zynq-Linux移植学习笔记之68- 国产ZYNQ添加用户自定义版本信息

1、背景介绍 在使用复旦微zynq时&#xff0c;有时候虽然针对uboot源码进行了改动&#xff0c;但由于uboot基线版本只有一个&#xff08;2018-07-fmsh&#xff09;&#xff0c;导致无法区分版本信息&#xff0c;虽然可以通过编译时间来区分&#xff0c;但没有版本号直观。内核也…

【Numpy】深入解析numpy中的ravel方法

NumPy中的ravel方法&#xff1a;一维化数组的艺术 &#x1f308; 欢迎莅临我的个人主页&#x1f448;这里是我深耕Python编程、机器学习和自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;并乐于分享知识与经验的小天地&#xff01;&#x1f387; &#x1f393; 博主简…

香港服务器负载过高的原因和应对办法

保持网站正常运行看似简单&#xff0c;但事实上&#xff0c;有许多问题会影响网站和应用程序的性能&#xff0c;并可能导致停机。其中一个问题就是服务器过载。而香港服务器作为一种常见的服务器类型&#xff0c;有时会出现负载过高的情况。为了帮助您确保在香港服务器过载不会…

AI应用案例:电能量异常分析智能诊断系统

窃电和计量装置故障造成漏收、少收电费使电力系统利益受损。一般情况主要通过定期巡检、定期校验电表、用户举报窃电等手段来发现窃电或计量装置故障。对人的依赖性太强&#xff0c;抓窃查漏的目标不明确。利用电力系统中逐步积累下来的海量真实数据&#xff0c;采用数据挖掘技…

C++多生产者,多消费者模型

C11实现多生产者&#xff0c;多消费者模型 在C标准库中实现多生产者多消费者模型&#xff0c;可以使用std::thread、std::queue、互斥锁(std::mutex)、条件变量(std::condition_variable)等组件。下面是一个简单的示例&#xff0c;展示如何创建多生产者和多消费者模型&#xf…

构建智能化的语言培训教育技术架构:挑战与机遇

随着全球化的发展和人们对语言学习需求的增长&#xff0c;语言培训教育行业正面临着越来越多的挑战和机遇。在这个背景下&#xff0c;构建智能化的语言培训教育技术架构成为提升服务质量和效率的重要手段。本文将探讨语言培训教育行业的技术架构设计与实践。 一、智能化教学平台…

Jupyter Notebook的三个使用场景:网页端、PyCharm专业版和VScode

说明&#xff0c;以下都是我个人的摸索感悟和总结&#xff0c;自己理解和猜测的是这样&#xff0c;欢迎指正。 Jupyter Notebook的三个常用使用地方&#xff08;网页端、PyCharm专业版、VScode&#xff09;&#xff1a; 总结一句话&#xff1a;网页端、PyCharm中和VScode中三…

Python使用multiprocessing实现多进程

大家好&#xff0c;当我们工作中涉及到处理大量数据、并行计算或并发任务时&#xff0c;Python的multiprocessing模块是一个强大而实用的工具。通过它&#xff0c;我们可以轻松地利用多核处理器的优势&#xff0c;将任务分配给多个进程并同时执行&#xff0c;从而提高程序的性能…

ROCm上情感分析:使用循环神经网络

15.2. 情感分析&#xff1a;使用循环神经网络 — 动手学深度学习 2.0.0 documentation (d2l.ai) 代码 import torch from torch import nn from d2l import torch as d2lbatch_size 64 train_iter, test_iter, vocab d2l.load_data_imdb(batch_size)class BiRNN(nn.Module):…

躺赚零撸项目,看广告赚红包,零门槛提现,秒到账,单机每日100+

这个项目是跟广告商直接对接的&#xff0c;跟以前小游戏看广告差不多&#xff0c;看完广告得金币5000个兑换5毛钱。 不过这个是可以直接提现&#xff0c;而是无门槛就可以提&#xff0c;有设备就可以操作&#xff0c;有空边看连续剧边刷也是挺香的&#xff0c;单机可以达到100…

组网智能是啥?

组网智能是一种基于穿透技术的远程连接解决方案&#xff0c;它为用户提供了操作简单、跨平台应用、无网络要求和独创的安全加速方案等优势。由于这些特点&#xff0c;组网智能已经被几十万用户广泛应用&#xff0c;解决了各行业客户的远程连接需求。 跨平台应用 组网智能具备跨…

《我的阿勒泰》观后感(二、返璞归真也是一种美)

看了李娟的小说《我的阿勒泰》逐渐悟到一个道理&#xff0c;返璞归真也是一种美&#xff0c;没必要每个人的人生三十年的年华&#xff0c;都去追求房子&#xff0c;车子等逐渐贬值的东西。人究竟应该追求怎样的一种活法&#xff1f; 什么是城市化&#xff1f;这是我听到的最好…

ffmpeg-webrtc(metartc)给ffmpeg添加webrtc协议

这个是使用metrtc的库为ffmpeg添加webrtc传输协议&#xff0c;目前国内还有一个这样的开源项目&#xff0c;是杨成立大佬&#xff0c;大师兄他们在做&#xff0c;不过wili页面维护的不好&#xff0c;新手不知道如何使用&#xff0c;我专门对它做过介绍&#xff0c;另一篇博文&a…