java的设计模式 - Builder模式
Builder 模式的目的?
构造对象的方式过于复杂,不如将之抽离出来。比如,构造器参数过多
这样说也有点抽象,举个例子吧。
举个例子
比如 非常热门的消息队列RabbitMQ 的 AMQP.BasicProperties
因为它的属性比较多,所以构造函数也是挺吓人的。
我看到也不太想调用。
如果现在要构造一条消息
- 投递模式(delivery mode)为 2
- 优先级(priority)是 2
- content-type 为 text/plain
在没有 builder 模式之前,是这样构造的
new AMQP.BasicProperties("text/plain",null,null,2,1,null,null,null,null,null,null,null,null,null);
痛苦啊!!!不信,你自己也可以尝试构造一下。
- 构造函数有很多你不想设置的参数
- 你要看准,哪个参数要赋值,哪个参数不赋值,一不小心就可能出错了。而这里有 14 个参数。。。
- 维护性差,写完代码再看一下,也看不出这个参数究竟是什么意思。还要点进去,一个一个参数地看才知道是什么意思
而用了 Builder 模式后。
new AMQP.BasicProperties.Builder()
.contentType("text/plain")
.deliveryMode(2)
.priority(1)
.build();
舒畅!!!
Builder 是如何实现?
很简单。
- 在
BasicProperties
中添加一个叫Builder
的内部类 - Builder 中所有字段和
BasicProperties
类是完全一致的 - 而
Builder
实例在调用build
函数的时候,再调用BasicProperties
的构造函数构造对象。
代码如下
public static class BasicProperties{
private String contentType;
private String contentEncoding;
private Map<String,Object> headers;
private Integer deliveryMode;
private Integer priority;
//... 还有很多属性
public BasicProperties(
String contentType,
String contentEncoding,
Map<String,Object> headers,
Integer deliveryMode,
//...
String clusterId){
this.contentType = contentTypel;
this.contentEncoding = contentEncoding;
//...
}
public static final class Builder {
private String contentEncoding;
private Map<String,Object> headers;
private Integer deliveryMode;
private Integer priority;
//.. 和BasicProperties的字段一致的。
public Builder contentType(String contentType){
this.contentType = contentType;
return this;
}
public Builder contentEncoding(String contentEncoding){
this.contentEncoding = contentEncoding;
return this;
}
public BasicProperties build() {
return new BasicProperties
( contentType
contentEncoding,
//还有很多属性
);
}
}
}
分析
Builder 模式的好处
- 不用花太多心思去记构造器的顺序,在 ide 中输入一个点就有自动提示了
- 好维护,很容易看到看明白这是什么属性
坏处
- 构造对象就要先调用 Buidler 构造器,多了构造器的开销
- 类的关系变得复杂了
其他的做法
如果不用 Builder 模式,有其他的做法吗?
重叠构造器?
比如,上面的例子,我构造的消息只需 投递模式(delivery mode)、优先级(priority)、 content-type ,专门为这几个参数弄个专门的构造函数,可以吗?
调用就变成这样了。
new AMQP.BasicProperties("text/plain",2,1)
可以,
- 但依然不太好看。
- 如果有不同的需求,各种属性都排列组合一下也麻烦。
- 不实际,因为类字段的类型可能会是一样的,有些组合注定不行
javaBean 模式呢?
BasicProperties p = new AMQP.BasicProperties();
p.setContentType("text/plain");
p.setDeliveryMode(2);
p.setPriority(1);
在《effective java》中就探讨过这个可能,书中是这样说的
因为构造过程被分到几个调用中,在构造过程中 javaBean 可能处于不一致的状态。类无法仅仅通过检验构造器参数的有效性来保证一致性。试图使用处于不一致状态的对象,将导致失败,这种失败与包含错误的代码大相径庭,因此它调试起来十分困难。与此相关的另一点不足在于,javaBeans 模式阻止了把类做成不可变的可能,这需要程序员付出格外的努力来确保它的线程安全。
这话就有点摸不着头脑,什么意思
其实意思是大概的,
- javaBean 是构造与字段赋值分离的,有可能 线程 1 在给对象 Obj 赋值,还没有赋完成的时候,线程 2 就拿了 Obj 的值了,就不一致了
- 如果 Obj 的字段全都是 final 的,不会出现上面那种情况,但字段只能会通过构造函数赋值(builder 模式也行),不能使用 javaBeans 的 setXXX 函数赋值了。
- 对多线程要求的,比如是传给消息队列的对象,程序员要保证下线程安全。
- 这也是个一个开放开闭的问题,Javabean 这样的写法确实和完全开放没啥区别,如果字段确定下来不用改了就最好设为 final 。
以上
java的设计模式 - Builder模式的更多相关文章
- java设计模式--Builder模式
一.Builder模式 二.使用例子 三.Spring中的Builder模式 Builder模式,构建者.构造者模式,在<图解设计模式>中归为 生成实例 一栏,该模式用于组装具有复杂结构的 ...
- 设计模式-----Builder模式
前言 近日,看到Myabtis中组件中SqlSessionFactory由SqlSessionFactoryBuilder().build()生成时,且采用Builder模式,遂记录学习之. SqlS ...
- Java 之 设计模式——代理模式
设计模式——代理模式 一.概述 1.代理模式 (1)真实对象:被代理的对象 (2)代理对象:代理真实对象的 (3)代理模式:代理对象代理真实对象,达到增强真实对象功能的目的 二.实现方式 1.静态代理 ...
- android 开发设计模式---Builder模式
我们通过一个例子来引出Builder模式.假设有一个Person类,我们通过该Person类来构建一大批人,这个Person类里有很多属性,最常见的比如name,age,weight,height等等 ...
- 【java】使用Builder模式,轻松应对动态繁杂的方法参数
背景:在咱编写的图片处理模块里,针对加载这个方法,参数很多,如: /** * 加载图片,经过内存.磁盘.两层缓存如果还没找到,则走http访问网络资源 * @param url 地址 * @param ...
- 设计模式-Builder模式(创建型模式)
//以下代码来源: 设计模式精解-GoF 23种设计模式解析附C++实现源码 //Product.h #pragma once class Product { public: Product(); ~ ...
- Java中的Builder模式
package com.mc.bsfram.others.entity; public class Person { private String name; private String addre ...
- java的设计模式 - 外观模式(Facade)
目的 看脸模式目的很简单,就是给用户留个好印象,不想让用户关注系统中的具体细节,关注系统的外表(暴露出来的接口)就好了.一些 GUI 的菜单也好,SDK 也好或多或少也会用到这种思想.这更多的是一种思 ...
- Java与设计模式-策略模式
在实际开发中,可能会遇到这样一个情况,某一功能的实现分为多种算法,这些算法能够认定为策略,在实际操作时选择不同算法或策略进行操作得出终于结果.在实际生活中.这些样例也是举不胜举.比如.商场举行活动,满 ...
随机推荐
- 【转】干货,Kubernetes中的Source Ip机制。
准备工作 你必须拥有一个正常工作的 Kubernetes 1.5 集群,用来运行本文中的示例.该示例使用一个简单的 nginx webserver 回送它接收到的请求的 HTTP 头中的源 IP 地址 ...
- SpringBoot之旅第一篇-初探
一.SpringBoot是什么? 微服务,应该是近年来最火的概念,越来越多的公司开始使用微服务架构,面试中被问到的微服务的概率很高,不管对技术的追求,还是为了进更好的公司,微服务都是我们开发人员的必须 ...
- Asp.Net Core&Jenkins持续交付到Windows Server
对于Linux平台上的持续集成和持续交付可以使用Docker快捷的完成任务,但是Windows Server下却不好使用,一般来讲,windows Server下最简单的发布方式是直接手动打包发布或是 ...
- C#开发APP,ToolBar控件在Smobiler中的使用方式【附案例源码】——Smobiler移动开发平台
控件说明 底部工具栏控件. 效果演示 其他效果 该界面为仿淘宝UI制作的一个简单的UI模板,源码获取方式请拉至文章末尾. 特色属性 属性 属性说明 Direction(相对布局) 容器主轴方向. Fl ...
- SAP MM Storage Location Missing in MD04 Result?
SAP MM Storage Location Missing in MD04 Result? Today I received a ticket from business team, a user ...
- SAP HUM事务代码 HUMAT 之初探
SAP HUM事务代码 HUMAT 之初探 SAP菜单中,该事务代码在这里: 1)如下的inbound delivery号码, Document flow, 已经完成了PGR, 2)执行HUMAT,进 ...
- 清理buffer/cache/swap的方法梳理
一.缓存机制介绍 在Linux系统中,为了提高文件系统性能,内核利用一部分物理内存分配出缓冲区,用于缓存系统操作和数据文件,当内核收到读写的请求时,内核先去缓存区找是否有请求的数据,有就直接返回,如果 ...
- [计算机视觉]100行python实现摄像机偏移、抖动告警
背景 在实际项目中,利用深度学习在检测道路车辆并分析车辆行为时,需要按照事先规定的方法绘制检测区(包含道路方向.车道区域等).由于各种原因(人为.天气),获取视频数据的摄像角度容易偏移原来设定的位置, ...
- 巡风源码阅读与分析---Aider.py
之前写过一遍Aider.py,但只是跟着代码一顿阅读没有灵魂,所以重新对它阅读并重新写一遍. 描述 文件位置:aider/aider.py 是用来辅助验证的脚本 官方描述就一句话 代码阅读分析 这个脚 ...
- LeetCode算法题-Robot Return to Origin(Java实现)
这是悦乐书的第281次更新,第298篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第149题(顺位题号是657).在2D平面上有一个从位置(0,0)开始的机器人.给定其移 ...