当前位置: 首页 > news >正文

08.哲说建造者模式(Builder Pattern)

“The odds that we’re in ‘base reality’ is one in billions.” —— Elon Musk
这段话出自马斯克在2016年的一次演讲,“人类活在真实世界的几率,可能不到十亿分之一”。此言一出,可谓一石激起千层浪。有人嘲讽马斯克是“语不惊人死不休”,也有人对他的言论深信不疑,更多的人则是把这种言论当作茶余饭后消遣的谈资。在这里插入图片描述
笔者对于 “世界是否真的真实” 这一问题的结果并不狂热,但我对于马斯克说出这句话的时间点很感兴趣,我可以帮大家捋一下前后的时间线。
2016年,ChatGPT诞生,而马斯克正是OpenAI的联合创办人之一
2017年,ChatGPT正式推出
2018年,ChatGPT开始全面发展,在社区活跃度直线上升
2019年,OpenAI从非营利性组织转型为“利润上限(caped-profit)”公司后,马斯克离开
2020年至今,ChatGPT逐渐火爆出圈,可以说改变人们的生活方式去日不远

在这里插入图片描述
微妙吗?“日心说” 源于计算结果,马斯克的话是否也源于即有结果的合理推测呢?这个话题可能每个人都会有自己的看法。
“天下万物生于有,有生于无。”——《道德经·第四十章》


一言

建造者模式是一步步创建一个复杂的对象,它允许用户只通过指定复杂对象的类型和内容就可以构建他们而不需要知道内部具体的构建细节。


概述

“天下万物生于有,有生于无”。如果现在我告诉你,你可以创建一个世界,你是一个造物主,但是你要把自己藏起来,不能让这个世界的小生命知道你的存在,你会以什么样的逻辑来设计第一个模型?
有人说,我要信“码”由缰,懒得装;有人说,我要挖空细节演好上帝。
这其实就是设计模式有趣的地方,它松散无骨,语气平和,不按它的来这个世界的太阳照常升起,但按它的要求做,这个世界会和谐的像假的一样。
这一次,我想让阅读我博客的你和我一起演一次“造物主”,从神的角度,俯视设计模式中的“建造者模式”。

一个看似无关紧要且过分晦涩的概念

建造者模式(Builder Pattern) 又叫生成器模式,是一种对象构建模式。它可以将复杂对象的建造过程抽象出来(抽象类别),使这个抽象过程的不同实现方法可以构造出不同表现 (属性) 的对象。


创世纪

好了,上帝们,我们开始。
我们假定创造一个世界的过程是:创建星球环境,播种生命,注入意识。但是每个世界肯定有不同的差别(作为上帝,我们不可能只创建了一个世界对不?)。那么自然有的世界的星球环境是单星结构,而有的是三体结构(很熟悉吧);生命形态肯定也不尽相同,碳基生物还是硅基生物或者其它神奇的生命体;意识就更是千变万化了。
在这里插入图片描述


信“码”由缰的上帝

如果我是那个只想快速完成KPI,不考虑运行后果的上帝,我可能会这样设计。
在这里插入图片描述

代码实现

一个抽象

public abstract class AbstractHouse {//基础环境public abstract void buildEarth();//创造生命public abstract void humanBeing();//意识觉醒public abstract void aware();//建造public void build(){buildEarth();humanBeing();aware();}
}

实现人类文明

public class HumanWorld extends AbstractHouse{@Overridepublic void buildEarth() {System.out.println("太阳系三号行星已构建");}@Overridepublic void humanBeing() {System.out.println("灵长类动物已生成");}@Overridepublic void aware() {System.out.println("人类意识觉醒,温良恭俭让");}
}

实现外星文明

public class ETWorld extends AbstractHouse{@Overridepublic void buildEarth() {System.out.println("猎户座九号行星已构建");}@Overridepublic void humanBeing() {System.out.println("硅基生物已生成");}@Overridepublic void aware() {System.out.println("异形意识觉醒,碾碎他们");}
}

经典反思

有什么问题?
优点自不必多说,两分钟的设计难度简单,易操作。
问题就是这个设计结构太简单了,上帝不会这么简单的去创建一个世界的。
没有任何的缓存层对象,程序可以说几乎没有可扩展和维护的空间。把产品(世界)和创建产品的过程(创世过程)完全封装在一起,耦合度太高了。

那么我们如何化腐朽为神奇呢?这就引出了建造者模式。


神之一手

先来了解下建造者模式的四个角色

  1. Product (产品角色) : 一个具体的产品对象;
  2. Builder (抽象建造者): 创建一个Product对象的各个部件指定的接口/抽象类;
  3. ConcreteBuilder (具体建造者):实现接口,构建和装配各个部件;
  4. Director (指挥者):构建一个Builder接口的对象,它主要是用于创建一个复杂的对象。它主要有两个作用,一是:隔离了客户与对象的生产过程;二是:负责控制产品对象的生产过程。

图解

在这里插入图片描述

代码

产品类

public class World{private String space;//环境private String alive;//生命private String aware;//意识//setter&getter
}

抽象建造者

public abstract class WorldBuilder {protected  World world = new World();//将建造的流程写好,抽象的方法public abstract void buildEarth();public abstract void humanBeing();public abstract void aware();//建造世界,将产品返回public World build(){return world;}
}

地球文明建造者

public class EarthBuilder extends WorldBuilder{@Overridepublic void buildEarth() {System.out.println("太阳系三号行星");world.setSpace("地球");}@Overridepublic void humanBeing() {System.out.println("地球生命");world.setAlive("人类");}@Overridepublic void aware() {System.out.println("地球人意识");world.setAware("温良恭俭让");}
}

外星文明建造者

public class ETBuilder extends WorldBuilder{@Overridepublic void buildEarth() {System.out.println("猎户座九号行星");world.setSpace("赛博坦");}@Overridepublic void humanBeing() {System.out.println("外星生命");world.setAlive("硅基生物");}@Overridepublic void aware() {System.out.println("外星人意识");world.setAware("活下去,撕碎他们");}
}

指挥者

public class WorldDirector {WorldBuilder worldBuilder = null; //方式1:构造器传入 worldBuilderpublic WorldDirector (WorldBuilder worldBuilder ) {this.worldBuilder = worldBuilder ;}//方式2:通过setter 传入 worldBuilderpublic void setWorldBuilder(WorldBuilder worldBuilder ) {this.worldBuilder = worldBuilder ;}//如何建造世界的流程,交给指挥者//创世public World constructWorld(){worldBuilder.buildEarth();worldBuilder.humanBeing();worldBuilder.aware();return worldBuilder.build();}
}

测试类

public class Client {public static void main(String[] args) {WorldDirector worldDirector = new WorldDirector (new EarthBuilder());World earth = worldDirector.constructWorld();worldDirector.setWorldBuilder(new ETBuilder());World et = worldDirector.constructWorld();System.out.println("地球人基本意识:"+earth.getAware());System.out.println("外星人基本意识:"+et.getAware());}
}

测试结果

在这里插入图片描述


Builder Pattern在JDK源码中的应用

其实设计模式离我们的日常开发非常近,建造者模式也不例外。比如StringBuilder,从名字看也差不多能猜到coder的思想和意图。
在这里插入图片描述
通过阅读Appendable、AbstractStringBuilder和StringBuilder的源码,我们会发现:

  • Appendable 接口定义了多个append方法(抽象方法),即Appendable 为抽象建造者,定义了抽象方法
  • AbstractStringBuilder 实现了 Appendable 接口方法,这里的AbstractstringBuilder 已经是建造者,只是不能实例化
  • StringBuilder 即充当了指挥者角色,同时充当了具体的建造者,建造方法的实现是由 AbstractStringBuilder 完成,而StringBuilder 继承了AbstractStringBuilder

客户端(使用程序)不必知道产品内部组成的细节,将产品本身与产品的创建过程解耦,使得相同的创建过程可以创建不同的产品对象每一个具体建造者都相对独立,而与其他的具体建造者无关,因此可以很方便地替换具体建造者或增加新的具体建造者,用户使用不同的具体建造者即可得到不同的产品对象。可以更加精细地控制产品的创建过程。将复杂产品的创建步骤分解在不同的方法中,使得创建过程更加清晰,也更方便使用程序来控制创建过程。增加新的具体建造者无须修改原有类库的代码,指挥者类针对抽象建造者类编程系统扩展方便,这也符合此前我们强调过的“开闭原则”。建造者模式所创建的产品一般具有较多的共同点,其组成部分相似,如果产品之间的差异性很大,则不适合使用建造者模式,因此其使用范围受到一定的限制。如果产品的内部变化复杂,可能会导致需要定义很多具体建造者类来实现这种变化,导致系统变得很庞大,因此在这种情况下,要考虑选择建造者模式是不是真的是最优解。
其实可以着重理解一下Director的构造这部分,在我看来,Builder Pattern的巧妙之处就在于它优雅的将一颗“种子”丢到了一个星球上,在“种子”发芽之前我们无需关注这颗种子的大小、品相。“万物生于有,有生于无”。

相信能耐心读到这里的同学应该会有一种奇怪的感觉,建造者模式怎么有点类似于抽象工厂模式呢?

那么抽象工厂模式是什么?工厂模式又是什么?它们和建造者模式又有什么异同呢?卖个关子,明年(下周)讲。

预祝大家新年快乐哈~


关注我,共同进步,每周至少一更,来聊的不只是代码。——Wayne

相关文章:

  • docker安装Nacos和Rabbitmq
  • 基于轻量级GhostNet模型开发构建生活场景下生活垃圾图像识别系统
  • 【Android12】Android Framework系列---tombstone墓碑生成机制
  • 【HarmonyOS】鸿蒙开发简介与项目基础配置演示
  • 编程笔记 html5cssjs 014 网页布局框架
  • Scikit-Learn线性回归(一)
  • 数据挖掘 聚类度量
  • macos下转换.dmg文件为 .iso .cdr文件的简单方法
  • 虚拟化技术和云计算的关系
  • CSS之元素转换
  • 【Flink-Kafka-To-Mysql】使用 Flink 实现 Kafka 数据写入 Mysql(根据对应操作类型进行增、删、改操作)
  • Java关键字(1)
  • MathType 运行时错误‘53’:文件未找到:MathPage.WLL_文件未找到mathpage.wll
  • 【Matlab】CNN卷积神经网络时序预测算法
  • 八皇后问题(C语言)
  • [ JavaScript ] 数据结构与算法 —— 链表
  • 【140天】尚学堂高淇Java300集视频精华笔记(86-87)
  • CAP理论的例子讲解
  • Java 23种设计模式 之单例模式 7种实现方式
  • java第三方包学习之lombok
  • Js基础知识(一) - 变量
  • leetcode378. Kth Smallest Element in a Sorted Matrix
  • Linux Process Manage
  • mysql 5.6 原生Online DDL解析
  • PHP 程序员也能做的 Java 开发 30分钟使用 netty 轻松打造一个高性能 websocket 服务...
  • Rancher-k8s加速安装文档
  • spring boot 整合mybatis 无法输出sql的问题
  • SQLServer插入数据
  • tensorflow学习笔记3——MNIST应用篇
  • 分布式事物理论与实践
  • 关于extract.autodesk.io的一些说明
  • 汉诺塔算法
  • 欢迎参加第二届中国游戏开发者大会
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • 推荐一款sublime text 3 支持JSX和es201x 代码格式化的插件
  • 以太坊客户端Geth命令参数详解
  • 用简单代码看卷积组块发展
  • 智能合约Solidity教程-事件和日志(一)
  • Mac 上flink的安装与启动
  • ( 10 )MySQL中的外键
  • (附源码)spring boot火车票售卖系统 毕业设计 211004
  • (附源码)ssm基于jsp高校选课系统 毕业设计 291627
  • (三维重建学习)已有位姿放入colmap和3D Gaussian Splatting训练
  • (续)使用Django搭建一个完整的项目(Centos7+Nginx)
  • (转)如何上传第三方jar包至Maven私服让maven项目可以使用第三方jar包
  • (总结)Linux下的暴力密码在线破解工具Hydra详解
  • .NET Core使用NPOI导出复杂,美观的Excel详解
  • .NET 中各种混淆(Obfuscation)的含义、原理、实际效果和不同级别的差异(使用 SmartAssembly)
  • .NETCORE 开发登录接口MFA谷歌多因子身份验证
  • .net实现头像缩放截取功能 -----转载自accp教程网
  • .vimrc php,修改home目录下的.vimrc文件,vim配置php高亮显示
  • [C++]18:set和map的使用
  • [Dxperience.8.*]报表预览控件PrintControl设置
  • [git]git命令如何取消先前的配置
  • [Linux]知识整理(持续更新)