首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >零件组装技术——建造者模式深度解析

零件组装技术——建造者模式深度解析

作者头像
文彬
发布于 2018-05-08 10:28:17
发布于 2018-05-08 10:28:17
75000
代码可运行
举报
文章被收录于专栏:醒者呆醒者呆
运行总次数:0
代码可运行

建造者模式是最后一个创建型设计模式,也是研究对象的创建。

将一个复杂对象的创建与它的表示分离,使得同样的构建过程可以创建不同的表示。

创建和表示是什么意思

表示就是外在,对象具体的样子,而内部构建过程是一种组装的概念,有点像套娃,同样的结构,但是却产生了不同大小的样子。

按照惯例,先讲故事。

假设要生产一部iPhone和一部ipod。我们要怎么做?

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class IPhone {
    private String camera;
    private String touchScreen;
    private String communication;
    //省略getter,setter方法。
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class Ipod {
    private String camera;
    private String touchScreen;
    //省略getter,setter方法。
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static void main(String[] args) {
    IPhone iPhone = new IPhone();
    Ipod ipod = new Ipod();
    iPhone.setCamera("1200 pixel");
    iPhone.setTouchScreen("retina");
    iPhone.setCommunication("TDMA");
    ipod.setCamera("800 pixel");
    ipod.setTouchScreen("NOVA");
}

我们知道同为apple旗舰产品的iPhone和ipod其实差不太多,只是iPhone的配置相对高一些,同时多了通讯模块。那么上面的代码是否显得臃肿,没有设计,因为都是在创建对象的时候去设置其属性,而且这两款产品的属性似乎差不多,好像可以套用一样的生产线。

建造者模式 = buildX() + construct() + (optional)ifX()

第一次建造

增加一个基类Apple产品类,包含并集的所有零件的属性。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class Apple {
    private String camera;
    private String touchScreen;
    private String communication;
    //省略getter,setter方法。

增加一个abstract AppleBuilder类,包含Apple的成员对象。然后加入buildX方法,以及钩子方法ifX用来判断某些只有一些子类特有的属性。最后要加一个成员对象apple的获取方法。

注意:这个apple对象要定义为protected的原因是它的子类要直接使用这个对象。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public abstract class AppleBuilder {
    protected Apple apple = new Apple();

    public abstract void buildCamera();

    public abstract void buildTouchScreen();

    public abstract void buildCommunication();

    public boolean ifCommunication() {
        return false;
    }

    public Apple createApple() {
        return apple;
    }
}

然后修改IPhone类和Ipod类,让他们实现AppleBuild类,可以直接使用基类的apple对象,而不必自身再去定义Apple中已经定义好的属性们,只需实现AppleBuilder类的abstract funciton buildX们。同时,不要忘记,由于默认ifCommunication是false,所以IPhone类一定要重写改钩子方法,修改为return true的方式。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class IPhone extends AppleBuilder {

    @Override
    public void buildCamera() {
        apple.setCamera("1200 pixel");
    }

    @Override
    public void buildTouchScreen() {
        apple.setTouchScreen("retina");
    }

    @Override
    public void buildCommunication() {
        apple.setCommunication("TDMA");
    }

    @Override
    public boolean ifCommunication() {
        return true;
    }

}
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class Ipod extends AppleBuilder {

    @Override
    public void buildCamera() {
        apple.setCamera("800 pixel");
    }

    @Override
    public void buildTouchScreen() {
        apple.setTouchScreen("NOVA");
    }

    @Override
    public void buildCommunication() {
        apple.setCommunication("none");
    }
}

最后,要加入关键的导演类,这里是车间类,用于真正的组装工作,对外提供装配方法construct();

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class Workshop {
    public static Apple construct(AppleBuilder ab) {
        ab.buildCamera();
        ab.buildTouchScreen();
        if (ab.ifCommunication()) {
            ab.buildCommunication();
        }
        Apple apple = ab.createApple();
        return apple;
    }
}

这样,就可以直接在客户端调用了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static void main(String[] args) {
    Apple apple = Workshop.construct(new IPhone());
    System.out.println("" + apple.getCamera());
}

客户端调用的时候,只需要新建一个workshop对象,然后调用其construct方法,传入具体的Apple产品子类即可获得一个Apple类。而new IPhone()我们可以使用配置文件的方式,这里给出代码。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static void main(String[] args) {
    Apple apple = Workshop.construct((AppleBuilder) XMLUtil.getBean());
    System.out.println("" + apple.getCamera());
}

配置文件

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<?xml version="1.0"?>
<config>
       <className>construction.IPhone</className>
</config> 

建立XMLUtil类

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class XMLUtil {
    public static Object getBean() {
        try {
            DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
            DocumentBuilder db = dbf.newDocumentBuilder();
            Document doc = db.parse(new File("E:\\Evsward\\workspace\\test\\src\\config.xml"));

            NodeList nl = doc.getElementsByTagName("className");
            Node n = nl.item(0).getFirstChild();
            String className = n.getNodeValue();

            Object c = Class.forName(className).newInstance();
            return c;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
}

以上就是第一次建造的全部代码了。我们可以清晰地看到完整的建造者模式的结构。几个要注意的点是:

  1. 目前的几个对象是可以抽象出来一个包含所有属性的对象,如此例中的Apple类。
  2. 我们去掉了具体的对象类,而是直接采用Builder的方式,将每个对象具体的内容实现在其里面。
  3. Builder要抽象出来一个基类,要包含上面的那个总对象以及该对象的对外获取方法。同时要注意设置该对象为protected,因为其子类Builder们要直接使用该对象,给该对象的属性赋值。
  4. 最重要的导演类,此例中的车间类,只提供一个construct方法,我觉得设置为static更好,外部可以直接通过类来调用。该方法内部要去调用具体的buildX的顺序。

第二次建造

这一次建造致力于最大限度精简化,此次建造属于探索性建造,不一定用于生产环境。

去掉导演类,将construct方法移入AppleBuilder类中。然后客户端调用方式为

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public static void main(String[] args) {
    Apple apple = AppleBuilder.construct((AppleBuilder) XMLUtil.getBean());
    System.out.println("" + apple.getCamera());
}

这样虽然精简结构了,但是会让代码变得不可读,AppleBuilder类的职责太多,不仅要创建对象,提供外部访问方法,还要声明各种buildX方法,以及可能出现的钩子方法,违背了“单一职责原则”。

所以当系统业务比较复杂的时候,不推荐省略导演类,完整的建造者模式会提高代码的可读性,以及更好的扩展。

适用场景

  1. 当要创建的对象内部属性比较复杂,且与其他对象有公共的部分的时候。然后他们的内部属性结构一定要稳定。
  2. 需要生成的对象属性可以变成buildX的形式,对属性赋值的顺序有要求。
    1. 隔离复杂对象的创建和使用,并使得相同的创建过程可以创建不同的产品。

创建过程是在导演类中进行,这就与使用隔离开来。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2017-09-13 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
建造者模式
某游戏软件公司决定开发一款基于角色扮演的多人在线网络游戏,玩家可以在游戏中扮演虚拟世界中的一个特定角色,角色根据不同的游戏情节和统计数据(例如力量、魔法、技能等)具有不同的能力,角色也会随着不断升级而拥有更加强大的能力。
千羽
2021/12/29
4550
建造者模式
每天5分钟-浅谈建造者模式
什么是建造者模式呢,顾名思义,把某个东西创建出来,说得准确点是,把各种部件按照需要的方式建造成一个产品。建造者模式「需要四个类,抽象建造者,具体建造者,产品类,导演类」。
用户8902830
2021/08/12
2670
每天5分钟-浅谈建造者模式
建造者模式(Builder)及其应用
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/gdutxiaoxu/article/details/52124383
程序员徐公
2018/09/18
1.6K0
建造者模式(Builder)及其应用
设计模式—— 十 一:建造者模式
建造者模式一步一步创建一个复杂的对象,它允许用户只通过指定复杂对象的类型和内容就 可以构建它们,用户不需要知道内部的具体构建细节。
三分恶
2020/07/16
8280
设计模式--建造者模式
建造者模式是一种对象创建型模式,它将复杂对象的构建与其表示分离,使得同样的构建过程可以创建不同的表示。它适用于构建复杂对象的场景,可以很好地解决“不同的对象需要复杂的组装过程”的问题。通过建造者模式,我们可以将对象的创建和表示分离,使得程序的扩展性和可维护性都得到了提高。
软件架构师Michael
2023/06/19
2560
设计模式 | 建造者模式及典型应用
建造者模式(Builder Pattern):将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。建造者模式是一种对象创建型模式。
小旋锋
2019/01/21
8690
设计模式----建造者模式
建造者模式(Builder Pattern):将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。建造者模式是一种对象创建型模式。
大忽悠爱学习
2021/11/15
3700
Java描述设计模式(06):建造者模式
一、生活场景 基于建造者模式,描述软件开发的流程。 1、代码实现 /** * 基于建造者模式描述软件开发 */ public class C01_InScene { public static void main(String[] args) { BuilderSoft builderSoft = new SoftImpl () ; ProjectManager manager = new ProjectManager(builderSoft) ;
知了一笑
2019/08/09
4840
一起学习设计模式--06.建造者模式
没有人买车会只买一个轮胎或一个方向盘,大家买的都是一辆包含轮胎、方向盘和发动机等多个部件的完整汽车。如何将这些部件组装成一辆完整的汽车并返回给用户,这是建造者模式需要解决的问题。
独立观察员
2022/12/06
3540
一起学习设计模式--06.建造者模式
Java设计模式(二)----建造者模式
建造者模式 一、概念 1、产品的内部表象 2、对象性质的建造 二、建造模式的结构 三、建造模式分成两个很重要的部分 四、使用场景 五、使用建造模式构建复杂对象 一、概念  建造模式是对象的创建模式。建造模式可以将一个产品的内部表象(internal representation)与产品的生产过程分割开来,从而可以使一个建造过程生成具有不同的内部表象的产品对象。 1、产品的内部表象   
汤高
2018/01/11
1.2K0
Java设计模式(二)----建造者模式
【Java设计模式】004-建造者模式
工厂类模式提供的是创建单个类的模式,而建造者模式则是将各种产品集中起来管理,用来创建复合对象,所谓复合对象就是指某个类具有不同的属性;
訾博ZiBo
2025/01/06
490
设计模式5之建造者模式
一个复杂的对象往往由多个子部件按一定的步骤组成。例如,汽车由发动机、轮胎、方向盘等部件组成的。
Lvshen
2022/05/05
1830
设计模式5之建造者模式
堆积木,建造者模式
假如一个对象的构建很复杂,需要很多步骤。则可以使用建造者模式,将其构建对象和组装成一个对象这两步给分开来。构建部分为(Builder)和组织部分(Director),实现了构建和装配的解耦。
BUG弄潮儿
2021/01/18
4100
堆积木,建造者模式
建造者模式【建造者模式设计模式】
指将一个复杂对象的构造与它的表示分离,使同样的构建过程可以创建不同的表示,这样的设计模式被称为建造者模式。它是将一个复杂的对象分解为多个简单的对象,然后一步一步构建而成。它将变与不变相分离,即产品的组成部分是不变的,但每一部分是可以灵活选择的。
高大北
2022/06/14
4980
深入理解建造者模式 ——组装复杂的实例
Java面试通关手册(Java学习指南,欢迎Star,会一直完善下去,欢迎建议和指导):https://github.com/Snailclimb/Java_Guide
用户2164320
2018/06/14
7550
深入理解建造者模式 ——组装复杂的实例
建造者模式的运用
* **分离**了部件的构造(由Builder来负责)和装配(由Director负责)。 从而可以构造出复杂的对象。这个模式适用于:某个对象的构建过程复杂的情况。
忆愿
2024/08/05
1080
建造者模式的运用
TS 设计模式02 - 建造者模式
工厂模式,为我们将客户端的生产行为封装起来,交给了工厂。它本质上是服务于客户端的,并没有降低产品生产的难度,产品的生产逻辑仍然在自己的类内部实现。 对于一些复杂的产品类(工序多,参数多),我们需要在内部维护其复杂的构建逻辑,是很容易出错的。 举一个简单的例子,生产牛肉汉堡,我们不管是由客户端去生产,还是工厂帮我们生产,建造的逻辑始终写在其 constructor 内部。全部生产步骤可能包含,做面包,做牛肉,放蔬菜,每个步骤可能有不同的参数控制,比如几片面包,几片牛肉或者几片蔬菜。如果我们发现之前的工序不好,需要调整工序,要么在类内部进行修改(违法开闭),要么新增一个类(成本太大,也不好维护),或者说我们要做猪肉汉堡,步骤和工序是一样的,我们新建一个类时由于步骤复杂,可能漏了或写错了。 那怎么办呢,建造者模式就是帮助我们创建一个复杂对象的,它将一个复杂的对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。比如我规定汉堡的构建工序是可以稳定地划分为,做面包,做肉,放蔬菜的,至于你具体是面包是做成方形的圆形的,肉要是什么类型,蔬菜是什么类型,这是具体的实现步骤,对不同的汉堡具体的实现不一样。
love丁酥酥
2020/07/28
7420
TS 设计模式02 - 建造者模式
掌握设计模式:深入理解建造者模式
设计模式是面向对象编程中的重要概念,它提供了一种解决常见问题的通用方法。在本文中,我们将深入探讨建造者模式,探讨它的定义、应用场景以及如何使用示例代码实现。
coderidea
2023/11/21
2820
掌握设计模式:深入理解建造者模式
Java设计模式之(三)——建造者模式
说人话:将构造复杂对象的过程和组成对象的部件解耦。就像攒电脑一样,不管什么品牌的配件,只要兼 容就可以装上;同样,一样的配件,可以有好多组装的方式。更直接点就是同一个类,通过内部不同属性状态构造不同的对象。
IT可乐
2021/11/22
4620
Java设计模式之(三)——建造者模式
建造者模式深度解析与实战应用
我是摘星,一名全栈开发者,专注 Java后端开发、AI工程化 与 云计算架构 领域,擅长Python技术栈。热衷于探索前沿技术,包括大模型应用、云原生解决方案及自动化工具开发。日常深耕技术实践,乐于分享实战经验与学习心得,希望用通俗易懂的方式帮助开发者快速掌握核心技术。持续输出AI、云计算及开源技术相关内容,欢迎关注交流!
摘星.
2025/06/09
1280
建造者模式深度解析与实战应用
相关推荐
建造者模式
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验