概念定义

Builder模式是一步一步创建一个复杂对象的创建型模式。该模式将构建复杂对象的过程和它的部件解耦,使得构建过程和部件的表示隔离开来。

应用场景

  • 对象创建过程比较复杂,或对创建顺序或组合有依赖(经典Builder模式,可参阅GOF《设计模式》)。
  • 创建对象时所需参数较多,且包含较多可选参数(变种Builder模式,可参阅《Effective Java》构建器小节)。

示例代码

本节侧重变种Builder模式,示例代码如下:

public class RobustPerson {
// 必要参数
private final int id; // 所有属性都被final修饰
private final String name; // 可选参数
private final int age;
private final String gender;
private final double height;
private final int weight; private RobustPerson(Builder builder) { // 构造方法私有,即客户端不能直接创建RobustPerson对象
this.id = builder.id;
this.name = builder.name;
this.age = builder.age;
this.gender = builder.gender;
this.height = builder.height;
this.weight = builder.weight;
} public static final class Builder {
// 必要参数
private final int id; // 必要属性被final修饰
private final String name; // 可选参数
private int age;
private String gender;
private double height;
private int weight; public Builder(int id, String name) { this.id = id; this.name = name; } // 必要参数通过构造方法赋值 public Builder age(int age) { this.age = age; return this; } // 可选参数通过同名方法赋值
public Builder gender(String gender) { this.gender = gender; return this; }
public Builder height(double height) { this.height = height; return this; }
public Builder weight(int weight) { this.weight = weight; return this; } public RobustPerson build() {
RobustPerson person = new RobustPerson(this);
// 复杂业务语义校验,对于校验不通过场景,抛出异常
if (person.height != 0 && person.weight != 0) { // Builder对象并非线程安全的,不能用this.xxx校验
double bmi = person.weight / (person.height * person.height);
if (bmi < 18 || bmi > 25) { // 身体质量指数(BMI)低于18或高于25时表示不健康
throw new IllegalArgumentException(person.name+" NOT A ROBUST PERSON!");
}
}
return person;
}
} public Builder toBuilder() { // 克隆
return new Builder(this.id, this.name).age(this.age)
.gender(this.gender).height(this.height).weight(this.weight);
} @Override
public String toString() {
return name + "{" + "id=" + id + ", age=" + age + ", gender='" + gender + '\'' +
", height=" + height + "m, weight=" + weight + "kg}";
}
}

测试类如下:

public class BuilderTest {
public static void main(String[] args) {
RobustPerson jack = new RobustPerson.Builder(1, "Jack")
.age(18).gender("male").height(1.70).weight(65).build();
System.out.println(jack); System.out.println("Jack keeps eating too much...");
System.out.println(jack.toBuilder().weight(80).build());
}
}

运行后输出:

Jack{id=1, age=18, gender='male', height=1.7m, weight=65kg}
Jack keeps eating too much...
Exception in thread "main" java.lang.IllegalArgumentException: Jack NOT A ROBUST PERSON!
at builder.RobustPerson$Builder.build(RobustPerson.java:48)
at builder.BuilderTest.main(BuilderTest.java:14)

关键特点

结合上节示例代码,可知Builder模式创建对象具有以下特点:

  • RobustPerson类的构造方法是私有的,即客户端不能直接创建RobustPerson对象。
  • RobustPerson类不可变(线程安全的): 所有属性都被final修饰,在构造方法中设置参数值,并且不对外提供Setter方法(Getter方法可选)。
  • 静态内部类Builder与RobustPerson拥有相同的成员变量,且Builder内通过构造方法处理final修饰的必要参数,通过同名方法处理可选参数。
  • Builder内的build()方法调用RobustPerson的私有构造函数来创建RobustPerson对象,且客户端只能通过该build()方法创建对象(从而避免Invalid状态)。
  • Builder对象并不具有线程安全性。如果需要对RobustPerson对象的参数强加约束条件,应对build()方法所创建的RobustPerson对象进行检验。
  • 当创建多个对象且对象大多数属性值都相同时,通过toBuilder()可简单高效地克隆对象,仅针对不同的属性重新设置值。
  • Builder模式使用链式调用,可读性更佳。

但Builder模式也不可避免地存在自身的缺点。例如:

  • 创建对象前必须先创建它的构建器,消耗内存(若仅需要链式调用可仿照Builder类定义目标类)。
  • Builder模式存在冗长的样板代码(可借助InnerBuilder或Lombok插件自动生成)。

业界实践

  • StringBuilder(JDK)
  • JobBuilder(quartz-2.3.0.jar)
  • SessionFactoryBuilder等(hibernate-core-5.3.6.Final.jar)

Java设计模式:Builder(构建器)模式的更多相关文章

  1. 深入探索Java设计模式之构建器模式(五)

    抽丝剥茧 细说架构那些事——[优锐课] 简单的程序不需要大量的设计过程,因为它们只关注有限的解决方案,仅使用几个类.大型程序专注于广泛的设计,该设计比好的设计范例的任何其他属性都更能利用可重用性.宏伟 ...

  2. Java设计模式系列-装饰器模式

    原创文章,转载请标注出处:<Java设计模式系列-装饰器模式> 一.概述 装饰器模式作用是针对目标方法进行增强,提供新的功能或者额外的功能. 不同于适配器模式和桥接模式,装饰器模式涉及的是 ...

  3. Java设计模式-Builder构造者模式

    介绍: 构造者模式,又称之为建造者模式,建造者模式,单例模式以及工厂模式都属于创建型模式1应用场景 今天学mybatis的时候,知道了SQLSessionFactory使用的是builder模式来生成 ...

  4. java设计模式之七装饰器模式(Decorator)

    顾名思义,装饰模式就是给一个对象增加一些新的功能,而且是动态的,要求装饰对象和被装饰对象实现同一个接口,装饰对象持有被装饰对象的实例,关系图如下: Source类是被装饰类,Decorator类是一个 ...

  5. java设计模式之 装饰器模式

    装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构. 这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装 ...

  6. Java设计模式之装饰器模式

    1.装饰器模式的定义(保持接口,扩展功能) Decorate装饰器,顾名思义,就是动态的给一个对象添加一些额外的职责,就好比对房子进行装修一样. 2.装饰器模式的特征 具有一个装饰对象. 必须拥有与被 ...

  7. java设计模式之装饰器模式以及在java中作用

    在JAVA I/O类库里有很多不同的功能组合情况,这些不同的功能组合都是使用装饰器模式实现的,下面以FilterInputStream为例介绍装饰器模式的使用  FilterInputStream和F ...

  8. java 设计模式 之 装饰器模式

    装饰器模式的作用 在不修改原先对象核心的功能的情况下,对功能进行增强. 增强对象的功能的途径 通过类继承的方式,对父对象进行增强操作,例如造车是父类,改装跑车,跑车加大灯,改装房车,房车加私人电影院. ...

  9. java设计模式4--建造者模式(Builder)

    本文地址:http://www.cnblogs.com/archimedes/p/java-builder-pattern.html,转载请注明源地址. 建造者模式 将一个复杂对象的构建与它的表示分离 ...

随机推荐

  1. PyCharm 快捷键失效解决办法

    PyCharm快捷键用着用着失效了 ......修改设置如下 重新启动Pycharm即可 原博客地址:https://blog.csdn.net/jacke121/article/details/82 ...

  2. Docker-compose安装和应用

    安装 1.为什么要使用docker-compose? 使用 Docker Compose 可以轻松.高效的管理容器,它是一个用于定义和运行多容器 Docker 的应用程序工具   2.其他前置条件需要 ...

  3. react官方脚手架添加less配置

    装两个包 npm install --save less less-loader 在node-modules/react-scripts/config/webpack.config.js中 在大概58 ...

  4. day99_12_3numpy的索引以及pandas的两个数据结构。

    一.索引与切片. nump的索引和python中的索引差不多,都是左开右闭区间. 如一个普通的array的索引,是由0开始的: res = np.array([1,2,3,4,5]) #### npa ...

  5. IntersectionObserver API,观察元素是否进入了可视区域

    网页开发时,常常需要了解某个元素是否进入了"视口"(viewport),即用户能不能看到它. 上图的绿色方块不断滚动,顶部会提示它的可见性. 传统的实现方法是,监听到scroll事 ...

  6. IT兄弟连 HTML5教程 CSS3揭秘 CSS常见的样式属性和值1

    CSS中的样式属性比较多,经常使用的属性可以分为这么几类:字体.文本.背景.位置.边框.列表,以及其他一些样式属性.每个类中的属性都可以单独使用:如果同一个类中多个属性一起使用,还可以将它们整合为一行 ...

  7. 损失函数--KL散度与交叉熵

    损失函数 在逻辑回归建立过程中,我们需要一个关于模型参数的可导函数,并且它能够以某种方式衡量模型的效果.这种函数称为损失函数(loss function). 损失函数越小,则模型的预测效果越优.所以我 ...

  8. 使用VBA从工作表中读图片,以及给工作表中写文件

    因为工作的原因,需要用到VBA,碰到读图片和写图片: Sub Macro01() '从工作表中保存图片 Application.ScreenUpdating = False Dim pth, shp, ...

  9. Web前端基础(10):JavaScript(四)

    1. 伪数组arguments arguments代表的是实参.有个讲究的地方是:arguments只在函数中使用. 1.1 返回参数个数 返回函数实参的个数:arguments.length 例子: ...

  10. kmeans均值聚类算法实现

    这个算法中文名为k均值聚类算法,首先我们在二维的特殊条件下讨论其实现的过程,方便大家理解. 第一步.随机生成质心 由于这是一个无监督学习的算法,因此我们首先在一个二维的坐标轴下随机给定一堆点,并随即给 ...