【设计模式】快速理解建造者模式,及其在JDK源码中的应用

简介: 建造者(Builder)模式在百度百科上的定义:是一种将复杂对象的构建和它的表示分离,使得同样的构建过程可以创建不同的表示。这段话在理解上十分抽象,简单来讲就是如果一个对象很复杂,使用建造者模式允许用户通过简单的方式构建这个对象,而不用关心对象具体构建的细节。且可以使用同样的构建过程可以创建出不同的对象。接下来将会通过具体的代码实例去讲解建造者模式。

(一)什么是建造者模式


建造者(Builder)模式在百度百科上的定义:是一种将复杂对象的构建和它的表示分离,使得同样的构建过程可以创建不同的表示。


这段话在理解上十分抽象,简单来讲就是如果一个对象很复杂,使用建造者模式允许用户通过简单的方式构建这个对象,而不用关心对象具体构建的细节。且可以使用同样的构建过程可以创建出不同的对象。


接下来将会通过具体的代码实例去讲解建造者模式。


(二)建造者模式中的几个角色


建造者模式中定义了四个主要角色,现在看定义可以不太容易理解,等结合之后的代码就容易理解了。


抽象建造者Builder


建造者的抽象类或者接口,主要用来定义具体建造者需要实现的一些方法。


具体建造者 ConcreteBuilder


具体的建造者,是抽象建造者的实现类。


具体的对象Client


最终想要生成的对象。


引导者Director


实例的生成需要依赖Director角色,Director生成实例不依赖ConcreteBuilder,它只通过调用Builder角色中定义的方法。


(三)建造者模式的实践


光看上面的概念比较难理解建造者模式,接下来通过一个造车的例子讲解建造者模式。

为了简化逻辑,这里将一辆车定义为由轮子、能源和颜色组成。首先定义出具体要生成的对象,也就是角色中的Client角色


publicclassCar {
// 轮子privateStringwheel;
// 能源privateStringenergy;
// 颜色privateStringcolor;
publicCar(){}
publicvoidsetColor(Stringcolor) {
this.color=color;
    }
publicvoidsetEnergy(Stringenergy) {
this.energy=energy;
    }
publicvoidsetWheel(Stringwheel) {
this.wheel=wheel;
    }
@OverridepublicStringtoString() {
return"Car{"+"wheel='"+wheel+'\''+", energy='"+energy+'\''+", color='"+color+'\''+'}';
    }
}

造车的逻辑可以抽象为造轮子,造能源,喷颜色,最后生成一辆车,因此可以将Builder角色构建出来

publicabstractclassBuilder {
publicabstractvoidbuildWheel();
publicabstractvoidbuildEnergy();
publicabstractvoidbuildColor();
publicabstractCargetCar();
}

具体每种车都有每种车的造法,因此分别构建两种车型的构建实体类,也就是角色中的ConcreteBuilder

publicclassBMWCarBuilderextendsBuilder {
privateCarcar;
publicBMWCarBuilder(){
car=newCar();
    }
@OverridepublicvoidbuildWheel() {
car.setWheel("四个轮子");
    }
@OverridepublicvoidbuildEnergy() {
car.setEnergy("油箱");
    }
@OverridepublicvoidbuildColor() {
car.setColor("黑色");
    }
@OverridepublicCargetCar() {
returncar;
    }
}

另一种具体建造者:

publicclassTeslaCarBuilderextendsBuilder {
privateCarcar;
publicTeslaCarBuilder(){
car=newCar();
    }
@OverridepublicvoidbuildWheel() {
car.setWheel("四个轮子");
    }
@OverridepublicvoidbuildEnergy() {
car.setEnergy("电池");
    }
@OverridepublicvoidbuildColor() {
car.setColor("白色");
    }
@OverridepublicCargetCar() {
returncar;
    }
}

最后需要有一个引导者Director角色,通过引导者调用Builder中定义的方法。

publicclassDirector {
privateBuilderbuilder;
publicDirector(Builderbuilder){
this.builder=builder;
    }
publicvoidbuildCar(){
builder.buildWheel();
builder.buildColor();
builder.buildEnergy();
    }
}

使用方式如下:首先创建了TeslaCarBuilder实例对象,交给引导者,引导者角色调用buildCar方法后TeslaCarBuilder就可以通过调用getCar获取对应的Car了。BMWCarBuilder也是同样的方式。

publicclassMain {
publicstaticvoidmain(String[] args) {
TeslaCarBuilderteslaCarBuilder=newTeslaCarBuilder();
Directordirector=newDirector(teslaCarBuilder);
director.buildCar();
CarteslaCar=teslaCarBuilder.getCar();
System.out.println(teslaCar.toString());
BMWCarBuilderbmwCarBuilder=newBMWCarBuilder();
director=newDirector(bmwCarBuilder);
director.buildCar();
CarbmwCar=bmwCarBuilder.getCar();
System.out.println(bmwCar.toString());
    }
}

还记得最开始对建造者模式的定义吗?使用建造者模式允许用户通过简单的方式构建这个对象,而不用关心对象具体构建的细节。且可以使用同样的构建过程可以创建出不同的对象。


在真正创建对象的过程中,已经不需要关注Car是如何build的,因此具体构建的细节都由各自的ConcreteBuilder去实现。并且同样的构建过程只要传入的ConcreteBuilder不同就可以创建不同的对象。


(四)建造者模式的变通


如果将具体建造者ConcreteBuilder、引导者Director、对象Client三种角色合为一种,那么建造者模式就会变得更加精简: 首先还是要定义抽象方法Builder

publicabstractclassBuilder {
publicabstractBuilderbuildWheel(Stringwheel);
publicabstractBuilderbuildEnergy(Stringenergy);
publicabstractBuilderbuildColor(Stringcolor);
publicabstractCargetCar();
}

使用时就更加方便了:

publicclassMain {
publicstaticvoidmain(String[] args) {
Carcar=newCar.CarBuilder()
                .buildColor("白色")
                .buildEnergy("新能源")
                .buildWheel("四个轮子")
                .getCar();
System.out.println(car.toString());
    }
}

(五)建造者模式在源码中的应用


StringBuilder就是建造者模式精简后的用法,StringBuilder继承了Appendable接口:

publicinterfaceAppendable {
Appendableappend(CharSequencecsq) throwsIOException;
Appendableappend(CharSequencecsq, intstart, intend) throwsIOException;
Appendableappend(charc) throwsIOException;
}

其具体的append方法和我们在第四章讲解的十分相似:

publicfinalclassStringBuilderextendsAbstractStringBuilderimplementsjava.io.Serializable, CharSequence{
@OverridepublicStringBuilderappend(Stringstr) {
super.append(str);
returnthis;
    }
}

在使用时,也是通过一路append实现链式编程。

stringBuilder.append("hello").append("world");

(六)总结


最传统的建造者模式体现着设计模式中“可替换性”的思想,Director不知道自己使用的具体是哪个Builder的实现类,所以Builder的实现类具备可替换性,也就可以理解为是一个个的组件。写代码时使用设计模式并不会让代码简单或者好写,反而会更复杂,但是在维护或者迭代时,就会省下很多精力。



相关文章
|
5天前
|
设计模式 安全 数据库连接
后端开发中的设计模式应用
在软件开发的浩瀚海洋中,设计模式如同灯塔,为后端开发者指引方向。它们不仅仅是代码的模板,更是解决复杂问题的智慧结晶。本文将深入探讨几种常见的设计模式,包括单例模式、工厂模式和观察者模式,并揭示它们在实际应用中如何提升代码的可维护性、扩展性和重用性。通过实例分析,我们将一窥这些模式如何在后端开发中大放异彩,助力构建高效、灵活的软件系统。
|
5天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
|
7天前
|
设计模式 数据库连接 PHP
PHP中的设计模式应用与最佳实践
在本文中,我们将探讨PHP设计模式的应用和最佳实践。通过深入分析,揭示如何在实际项目中有效利用设计模式来优化代码结构、提升系统灵活性和维护性,并分享一些常见设计模式的实际应用案例。无论你是PHP初学者还是经验丰富的开发者,这篇文章都会对你有所帮助。
|
25天前
|
设计模式 算法 开发者
深入理解工厂模式与策略模式:设计模式的灵活应用
深入理解工厂模式与策略模式:设计模式的灵活应用
|
3天前
|
设计模式 算法 PHP
PHP中的设计模式:策略模式的应用与实践
在软件开发中,设计模式是解决问题的最佳实践。本文将探讨PHP中的策略模式,通过实际应用案例,展示如何有效地使用策略模式来提高代码的灵活性和可维护性。我们将从基本概念入手,逐步深入到实际编码,最终实现一个具有策略模式的应用。
|
27天前
|
算法 安全 Java
深入JDK源码:揭开ConcurrentHashMap底层结构的神秘面纱
【8月更文挑战第24天】`ConcurrentHashMap`是Java并发编程中不可或缺的线程安全哈希表实现。它通过精巧的锁机制和无锁算法显著提升了并发性能。本文首先介绍了早期版本中使用的“段”结构,每个段是一个带有独立锁的小型哈希表,能够减少线程间竞争并支持动态扩容以应对高并发场景。随后探讨了JDK 8的重大改进:取消段的概念,采用更细粒度的锁控制,并引入`Node`等内部类以及CAS操作,有效解决了哈希冲突并实现了高性能的并发访问。这些设计使得`ConcurrentHashMap`成为构建高效多线程应用的强大工具。
33 2
|
19天前
|
前端开发 C# 设计模式
“深度剖析WPF开发中的设计模式应用:以MVVM为核心,手把手教你重构代码结构,实现软件工程的最佳实践与高效协作”
【8月更文挑战第31天】设计模式是在软件工程中解决常见问题的成熟方案。在WPF开发中,合理应用如MVC、MVVM及工厂模式等能显著提升代码质量和可维护性。本文通过具体案例,详细解析了这些模式的实际应用,特别是MVVM模式如何通过分离UI逻辑与业务逻辑,实现视图与模型的松耦合,从而优化代码结构并提高开发效率。通过示例代码展示了从模型定义、视图模型管理到视图展示的全过程,帮助读者更好地理解并应用这些模式。
34 0
|
19天前
|
设计模式 安全 数据库连接
|
21天前
|
设计模式 JavaScript 前端开发
从工厂到单例再到策略:Vue.js高效应用JavaScript设计模式
【8月更文挑战第30天】在现代Web开发中,结合使用JavaScript设计模式与框架如Vue.js能显著提升代码质量和项目的可维护性。本文探讨了常见JavaScript设计模式及其在Vue.js中的应用。通过具体示例介绍了工厂模式、单例模式和策略模式的应用场景及其实现方法。例如,工厂模式通过`NavFactory`根据用户角色动态创建不同的导航栏组件;单例模式则通过全局事件总线`eventBus`实现跨组件通信;策略模式用于处理不同的表单验证规则。这些设计模式的应用不仅提高了代码的复用性和灵活性,还增强了Vue应用的整体质量。
14 0
|
29天前
|
设计模式 SQL 缓存
Java编程中的设计模式:单例模式的深入理解与应用
【8月更文挑战第22天】 在Java的世界里,设计模式是构建可维护、可扩展和灵活的软件系统的基石。本文将深入浅出地探讨单例模式这一经典设计模式,揭示其背后的哲学思想,并通过实例演示如何在Java项目中有效运用。无论你是初学者还是资深开发者,这篇文章都将为你打开一扇洞悉软件设计深层逻辑的大门。
26 0