设计模式:Builder
简介
建造者模式(Builder),将一个复杂对象的表示和它的构建分离,这样同样的构造过程可以创建出不同的对象状态。
类图
下面的Product是要创建的对象的目标类型,产品。

Builder
创建一个Product对象涉及的操作的抽象接口,定义了Product各个部分的创建方法。Director
使用Builder来构建Product,控制构建过程。ConcreteBuilder
一个具体的构建者。
样例代码
目标类型Robot
假设要构建一个机器人Robot类型的对象:
public class Robot {
private String head;
private String arm;
private String leg;
private String body;
// getter & setter 省略
public String getDescription() {
return toString();
}
@Override
public String toString() {
return "Robot{" +
"head='" + head + '\'' +
", arm='" + arm + '\'' +
", leg='" + leg + '\'' +
", body='" + body + '\'' +
'}';
}
}
RobotBuilder
使用RobotBuilder接口(也可以是抽象类)来定义Robot各个部分的设置。
public interface RobotBuilder {
void setArm();
void setLeg();
void setBody();
void setHead();
}
具体的RobotBuilder
可以定义不同RobotBuilder子类来实现不同的Robot的构建。
BigRobotBuilder
构建一个“Big Robot”的建造器如下:
public class BigRobotBuilder implements RobotBuilder {
private Robot mRobot;
public BigRobotBuilder(Robot robot) {
mRobot = robot;
}
@Override
public void setArm() {
mRobot.setArm("Big arm");
}
@Override
public void setLeg() {
mRobot.setLeg("Big leg");
}
@Override
public void setBody() {
mRobot.setBody("Big body");
}
@Override
public void setHead() {
mRobot.setHead("Big head");
}
}
MiniRobotBuilder
构建一个“Mini Robot”的建造器如下:
public class MiniRobotBuilder implements RobotBuilder {
private Robot mRobot;
public MiniRobotBuilder(Robot robot) {
mRobot = robot;
}
@Override
public void setArm() {
mRobot.setArm("Mini arm");
}
@Override
public void setLeg() {
mRobot.setLeg("Mini leg");
}
@Override
public void setBody() {
mRobot.setBody("Mini body");
}
@Override
public void setHead() {
mRobot.setHead("Mini head");
}
}
RobotBuildDirector
控制Robot构建过程的指挥者:
public class RobotBuildDirector {
public void buildRobot(RobotBuilder builder) {
builder.setBody();
builder.setArm();
builder.setHead();
builder.setLeg();
}
}
如果构建过程需要严格顺序的话,那么Director来封装具体构建过程是很重要的。
调用代码
创建一个具体的builder,然后执行Director.buildRobot()来完成构建:
void main() {
Robot robot = new Robot();
RobotBuildDirector director = new RobotBuildDirector();
RobotBuilder builder = new BigRobotBuilder(robot);
director.buildRobot(builder);
// 显示已构建的robot的详细信息
Log.println(robot.getDescription());
}
小结
以上的Robot案例展示了builder模式的标准形式。
实际使用中,可能没有抽象Builder和Director控制过程这样的需要。
而是,仅仅希望将包含很多可设置的属性的类型的构建相关的代码与类型本身分开。
而且,设置属性的类往往就作为要配置的类的子类。
下面的实际案例中会说明这点。
实际案例
Diaglog
在Android开发中,AlertDialog类型的构建就使用到了Builder模式,而且是简化了的。
AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(this);
AlertDialog dialog = dialogBuilder.setIcon(R.drawable.ic_launcher)
.setMessage("message")
.setTitle("Title")
.setNegativeButton("Cancel", null)
.create();
内部类AlertDialog.Builder用来对AlertDialog的不同的属性进行设置。
注意到setMessage()、setTitle()等方法都返回AlertDialog.Builder对象本身,
这样就可以以“链式代码”的形式对多个属性进行设置。这样的api很常见,如:
StringBuilder sb = new StringBuilder();
sb.append(1).append(2).append(2);
再如:
SharedPreferences sp = getSharedPreferences("1", MODE_PRIVATE);
sp.edit().putBoolean("1", false)
.putInt("2", 2)
.putLong("3", 3L)
.remove("2")
.apply();
可以看到,Builder对外暴露实际要使用的AlertDialog的各种属性的设置,而且它可以提供一些默认配置。这样使用者就无需每个属性都去指定。
当调用者完成对AlertDialog的一些方面的定制后,执行create()返回最终的dialog对象。
其它例子
Notification也是:
Notification.Builder builder = new Notification.Builder(this);
Notification.Action action = null;
String category = "";
builder.addAction(action)
.setCategory(category)
.setColor(0xff00ff00)
.setNumber(10086)
.setGroup("groupKey")
.build();
其它一些第三方库,或者非Android框架中也有类似Builder模式的大量运用,贵在积累和学习。
总结
可以看到,实际使用中的builder模式和标准的模式组成稍微不同。
很多已知的API中,对构建着模式的使用就是:
将一个包含很多不同属性的对象的设置相关逻辑封装为一个静态内部类去完成。
(本文使用Atom编写)
设计模式:Builder的更多相关文章
- 设计模式Builder(建造者)模式
1.出现原因 在软件系统中,有时候会面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成:由于需求的变化,这个复杂的对象的各个部分可能面临着剧烈的变化,但是把他们组合在一起的算法 ...
- 设计模式-----Builder模式
前言 近日,看到Myabtis中组件中SqlSessionFactory由SqlSessionFactoryBuilder().build()生成时,且采用Builder模式,遂记录学习之. SqlS ...
- java设计模式--Builder模式
一.Builder模式 二.使用例子 三.Spring中的Builder模式 Builder模式,构建者.构造者模式,在<图解设计模式>中归为 生成实例 一栏,该模式用于组装具有复杂结构的 ...
- 设计模式-Builder和Factory模式区别
Builder和Factory模式区别 Builder模式结构: Factory模式一进一出,Builder模式是分步流水线作业.当你需要做一系列有序的工作或者按照一定的逻辑来完成创建一个对象时 Bu ...
- 设计模式-Builder模式(创建型模式)
//以下代码来源: 设计模式精解-GoF 23种设计模式解析附C++实现源码 //Product.h #pragma once class Product { public: Product(); ~ ...
- Java设计模式-Builder构造者模式
介绍: 构造者模式,又称之为建造者模式,建造者模式,单例模式以及工厂模式都属于创建型模式1应用场景 今天学mybatis的时候,知道了SQLSessionFactory使用的是builder模式来生成 ...
- 一天一个设计模式——Builder建造者模式
一.模式说明 在现实世界中,当我们要构造一个大型工程时(建一个大楼),通常的做法是先建造工程的每个独立部分,然后再逐步构造完成(先打地基,再搭框架,最后逐层累造).在程序设计领域,构造一个复杂的类时( ...
- C++设计模式-Builder建造者模式
作用:将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. Builder模式和AbstractFactory模式在功能上很相似,因为都是用来创建大的复杂的对象,它们的区别是:B ...
- Java设计模式-Builder生成器模式
概念: 生成器模式也称之为建造者模式.生成器模式的意图在于将一个复杂的构建与其表示相分离,构建与产品分离. UML: Ibuild接口清晰地反映了创建产品Product的流程. 生成器模式涉及4个关键 ...
- java的设计模式 - Builder模式
Builder 模式的目的? 构造对象的方式过于复杂,不如将之抽离出来.比如,构造器参数过多 这样说也有点抽象,举个例子吧. 举个例子 比如 非常热门的消息队列RabbitMQ 的 AMQP.Basi ...
随机推荐
- Web应用获取文件路径的方法
拥有 HttpServletRequest req 对象 req.getSession().getServletContext().getRealPath("/") ----- ...
- C#中隐式运行CMD命令行窗口的方法
using System; using System.Diagnostics; namespace Business { /// <summary> /// Command 的摘要说明. ...
- 数的划分(NOIP2001&水题测试2017082401)
题目链接:数的划分 这题直接搜索就行了.给代码,思路没什么好讲的,要讲的放在代码后面: #include<bits/stdc++.h> using namespace std; int d ...
- NOIP水题测试(2017082401)
哈,水题测试又来了! 上次的水题简单吧! 答案是以单题形式发布的(旅行家的预算随后发布). 下面来看今天的题,还是水题. 时间限制:5小时 题目一:看上去就很水 题目二:比上面一题还水 题目三:数的划 ...
- Syslog和Windows事件日志收集
Syslog和Windows事件日志收集 EventLog Analyzer从分布式Windows设备收集事件日志,或从分布式Linux和UNIX设备.交换机和路由器(Cisco)收集syslog.事 ...
- ac自动机板子
hdu2222 #include<bits/stdc++.h> #define ll long long #define M 500005 using namespace std; int ...
- SQL2008 2机镜像
清除设置 //删除端点 declare @sql varchar() declare @mirrName varchar() select @mirrName=name from sys.databa ...
- 吓死baobao了
早上远程连接服务器连不上,后面重启服务器,显示进入紧急修复模式:welcome to emergency mode!after logging in ,type “journalctl -xb” to ...
- 第10章:MongoDB-CRUD操作--文档--修改--修改器
① $set:进行内容的重新设置 语法:{"$set" : {"成员" : "新内容"}}: 范例:将年龄是20岁的人的成绩修改为89 db ...
- Effective C++ 随笔(4)
条款21:必须返回对象时,别妄想返回其reference 例子: Raional类可以执行有理数的一些运算,并且使用heap内存申请 并且其operator*函数为 const Rational&am ...