Java设计中的Builder模式的介绍


Posted in Javascript onMarch 22, 2018

概述

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

使用场景

  • 相同的方法,不同的执行顺序,产生不同的事件结果
  • 多个部件或零件,都可以装配到一个对象中,但产生的运行结果又不相同时
  • 产品类非常复杂,或者产品类中的调用顺序不同产生不同的作用
  • 当初始化一个对象特别复杂,如参数多,且很多参数都具有默认值时

下面从一个Person例子进行分析重叠构造器模式、JavaBeans模式、Builder模式的使用。Person类有两个必要参数(id和name),有5个可选参数(age,sex,phone,address和desc)

重叠构造器模式

在这种模式下,你提供第一个只有必要参数的构造器,第二个构造器有一个可选参数,第三个有两个可选参数,依此类推,最后一个构造器包含所有的可选参数。下面看看其编程实现:

/ * 使用重叠构造器模式
 */
public class Person {
 //必要参数
 private final int id;
 private final String name;
 //可选参数
 private final int age;
 private final String sex;
 private final String phone;
 private final String address;
 private final String desc;
 public Person(int id, String name) {
 this(id, name, 0);
 }
 public Person(int id, String name, int age) {
 this(id, name, age, "");
 }
 public Person(int id, String name, int age, String sex) {
 this(id, name, age, sex, "");
 }
 public Person(int id, String name, int age, String sex, String phone) {
 this(id, name, age, sex, phone, "");
 }
 public Person(int id, String name, int age, String sex, String phone, String address) {
 this(id, name, age, sex, phone, address, "");
 }
 public Person(int id, String name, int age, String sex, String phone, String address, String desc) {
 this.id = id;
 this.name = name;
 this.age = age;
 this.sex = sex;
 this.phone = phone;
 this.address = address;
 this.desc = desc;
 }
}

从上面的代码中,当你想要创建实例的时候,就利用参数列表最短的构造器,但该列表中包含了要设置的所有参数:

Person person = new Persion(1, "李四", 20, "男", "18800000000", "China", "测试使用重叠构造器模式");

重叠构造器可行,但是当有许多参数的时候,创建使用代码会很难写,并且较难以阅读

JavaBeans模式

采用Javabean 的写法,写一堆属性的setter方法,通过生成对象,让后调用setter方法给属性赋值。

/ * 使用JavaBeans模式
 */
public class Person {
 //必要参数
 private int id;
 private String name;
 //可选参数
 private int age;
 private String sex;
 private String phone;
 private String address;
 private String desc;
 public void setId(int id) {
 this.id = id;
 }
 public void setName(String name) {
 this.name = name;
 }
 public void setAge(int age) {
 this.age = age;
 }
 public void setSex(String sex) {
 this.sex = sex;
 }
 public void setPhone(String phone) {
 this.phone = phone;
 }
 public void setAddress(String address) {
 this.address = address;
 }
 public void setDesc(String desc) {
 this.desc = desc;
 }

这种模式弥补了重叠构造器模式的不足。创建实例很容易,这样产生的代码读起来也很容易:

Person person = new Person();
person.setId(1);
person.setName("李四");
person.setAge(20);
person.setSex("男");
person.setPhone("18800000000");
person.setAddress("China");
person.setDesc("测试使用JavaBeans模式");

JavaBeans模式因为构造过程被分到了几个调用中,在构造过程中JavaBean可能处于不一致的状态。类无法仅仅通过检验构造器参数的有效性来保证一致性。

Builder模式

Builder模式既能保证像重叠构造器那样的安全,也能实现JavaBean模式那样的可读性。

/**
 * 使用Builder模式
 */
public class Person {
 //必要参数
 private final int id;
 private final String name;
 //可选参数
 private final int age;
 private final String sex;
 private final String phone;
 private final String address;
 private final String desc;
 private Person(Builder builder) {
 this.id = builder.id;
 this.name = builder.name;
 this.age = builder.age;
 this.sex = builder.sex;
 this.phone = builder.phone;
 this.address = builder.address;
 this.desc = builder.desc;
 }
 public static class Builder {
 //必要参数
 private final int id;
 private final String name;
 //可选参数
 private int age;
 private String sex;
 private String phone;
 private String address;
 private String desc;
 public Builder(int id, String name) {
  this.id = id;
  this.name = name;
 }
 public Builder age(int val) {
  this.age = val;
  return this;
 }
 public Builder sex(String val) {
  this.sex = val;
  return this;
 }
 public Builder phone(String val) {
  this.phone = val;
  return this;
 }
 public Builder address(String val) {
  this.address = val;
  return this;
 }
 public Builder desc(String val) {
  this.desc = val;
  return this;
 }
 public Person build() {
  return new Person(this);
 }
 }
}

Person是不可变得,所有的默认参数值都单独放在一个地方。builder的setter方法返回builder本身,以便可以链式调用。下面是客户端使用代码:

Person person = new Person.Builder(1, "张三")
                .age(18).sex("男").desc("测试使用builder模式").build();

Android源码中的Builder模式

  • AlertDialog源码
  • Universal-Image-Loader图片库
  • OkHttp的Request源码

优点:

  • 良好的封装性,使用建造者模式可以使客户端不比知道产品内部组成的细节
  • 建造者独立,容易扩展(可通过抽象———接口,抽象类)

缺点:

  • 会产生多余的Builder对象以及Director对象,消耗内存

总结 :Builder模式通常作为配置类的构建器将配置的构建和表示分离开来,同时也是将配置从目标类中隔离出来,避免过多的setter方法。

Javascript 相关文章推荐
js和as的稳定传值问题解决
Jul 14 Javascript
Js判断CSS文件加载完毕的具体实现
Jan 17 Javascript
javascript变量声明实例分析
Apr 25 Javascript
jquery对所有input type=text的控件赋值实现方法
Dec 02 Javascript
JavaScript中的call和apply的用途以及区别
Jan 11 Javascript
Angular浏览器插件Batarang介绍及使用
Feb 07 Javascript
vue props传值失败 输出undefined的解决方法
Sep 11 Javascript
layDate插件设置开始和结束时间
Nov 15 Javascript
vue-router 起步步骤详解
Mar 26 Javascript
laravel-admin 与 vue 结合使用实例代码详解
Jun 04 Javascript
vue实现前端列表多条件筛选
Oct 26 Javascript
jQuery实现简单弹幕制作
Dec 10 jQuery
使用vue-route 的 beforeEach 实现导航守卫(路由跳转前验证登录)功能
Mar 22 #Javascript
收集前端面试题之url、href、src
Mar 22 #Javascript
vue 的keep-alive缓存功能的实现
Mar 22 #Javascript
bootstrap中selectpicker下拉框使用方法实例
Mar 22 #Javascript
使用Vue开发动态刷新Echarts组件的教程详解
Mar 22 #Javascript
JavaScript分步实现一个出生日期的正则表达式
Mar 22 #Javascript
vuex 的简单使用
Mar 22 #Javascript
You might like
PHP SEO优化之URL优化方法
2011/04/21 PHP
ThinkPHP模板中数组循环实例
2014/10/30 PHP
PHP间隔一段时间执行代码的方法
2014/12/02 PHP
ThinkPHP实现ajax仿官网搜索功能实例
2014/12/02 PHP
php简单smarty入门程序实例
2015/06/11 PHP
PHP定时执行任务的3种方法详解
2015/12/21 PHP
深入剖析浏览器退出之后php还会继续执行么
2016/05/17 PHP
给大家分享几个常用的PHP函数
2017/01/15 PHP
利用javascript解决图片缩放及其优化的代码
2012/05/23 Javascript
Jquery事件的连接使用示例
2013/06/18 Javascript
JavaScript判断表单提交时哪个radio按钮被选中的方法
2015/03/21 Javascript
js获得当前系统日期时间的方法
2015/05/06 Javascript
jQuery实现带有动画效果的回到顶部和底部代码
2015/11/04 Javascript
简单谈谈ES6的六个小特性
2016/11/18 Javascript
原生JS和jQuery操作DOM对比总结
2017/01/19 Javascript
Angular.js中控制器之间的传值详解
2017/04/24 Javascript
详解vue express启动数据服务
2017/07/05 Javascript
Angular6封装http请求的步骤详解
2018/08/13 Javascript
vue打包使用Nginx代理解决跨域问题
2018/08/27 Javascript
微信小程序按钮点击动画效果的实现
2019/09/04 Javascript
node.js实现简单的压缩/解压缩功能示例
2019/11/05 Javascript
js中addEventListener()与removeEventListener()用法案例分析
2020/03/02 Javascript
vue 组件基础知识总结
2021/01/26 Vue.js
一个计算身份证号码校验位的Python小程序
2014/08/15 Python
python通过线程实现定时器timer的方法
2015/03/16 Python
Python实现竖排打印传单手机号码易撕条
2015/03/16 Python
Centos 升级到python3后pip 无法使用的解决方法
2018/06/12 Python
用python打开摄像头并把图像传回qq邮箱(Pyinstaller打包)
2020/05/17 Python
使用CSS3中的calc()属性来以算式表达尺寸数值
2016/06/06 HTML / CSS
html5音频_动力节点Java学院整理
2018/08/22 HTML / CSS
Html5新标签解释及用法
2012/02/17 HTML / CSS
演讲比赛的活动方案
2014/08/28 职场文书
公务员年终个人总结
2015/02/12 职场文书
违反纪律检讨书范文
2015/05/07 职场文书
优秀学生主要事迹怎么写
2015/11/04 职场文书
MySQL中你可能忽略的COLLATION实例详解
2021/05/12 MySQL