一.定义:
建造者模式(Builder Pattern):将一个复杂对象的构建与它的表示分离,是的同样的构建过程可以创建不同的表示。
建造者本质:对象的构建和装配分离:比如:属性当做组件在满足条件的时候进行装配
1. List item分离的对象子组件的单独构造(由Builder构造)和装配(有Director来负责)。从而可以构造出复杂的对象。这个模式适用于:某个对象的构造过程复杂。
2.由于实现了构建和装配的解耦。不同的构建器,相同的装配,也可以做出不同的对象。相同的构建器,不同的装配顺序也可以做出不同的对象。也就是实现了构建算法、装配算法的解耦,实现了更好的复用。
3.建造者模式可以将部件和其组装过程分开,一步一步创建一个复杂对象。用户只需要指定复杂对象的类型就可以得到该对象,而无需知其内部的具体构造细节。
4.与工厂模式的区别是:建造模式 更加关注零件装配的顺序
二.建造者中的角色
1.建造者(Builder):为创建一个产品对象的各个部件指定抽象接口。
2.具体建造者(ConcreteBuilder):实现Builder接口以构建和装配该产品的各个部件,定义并明确它所创建的表示,并提供一个检索产品接口。
3.指挥者(Director):指挥并构造一个使用Builder接口的对象
4.产品(Product):表示被构造的复杂对象。ConcreteBuilder创建该产品的内部表示并定义它的装配过程,包含定义组成部件的类,包括将这些部件装配成最终产品的接口。
具体的例子UML图
三.应用场景
1.java中的StringBuilder类的append方法;
2.SQL中的PreparedStatement;
3.JDOM中,DOMBuilder、SAXBuilder;
4.Mybatis中用到的建造者模式:
SqlSessionFactoryBuilder、XMLConfigBuilder、XMLMapperBuilder、XMLStatementBuilder、CacheBuilder等。
比如:XMLConfigBuilder读取配置文件构建出Configuration对象,然后SqlSessionFactoryBuiler使用Configuration对象作为参数,构建出SqlSessionFactory对象,
原因:mybatis的初始化比较复杂,不是一个构造函数就能包括的。所以采用分层构建。例如Mybatis中及其重要的Configuration对象,它庞大且复杂,初始化比较麻烦,所以使用专门的构造者XMLConfigBuilder进行构建。
5.一个对象有非常复杂的内部结构,想把复杂对象的创建和试用分离;
四.优缺点.
有点:
1.封装性好,创建和使用;无需修改原代码符合“开闭原则”。
2.拓展性好,建造者类之间独立,一定程度上解耦;代码解耦,模块化,方便维护。产品本身与创建过程解耦,可以使用相同的创建过程得到不动的产品。
确定:
1.产品必须有共同特点,使用范围有限。建造者模式创造出来的产品,其组成部分基本相同。如果产品之间的差异较大,则不适用这个模式。
2.若产品内部发生变化,建造者都要修改,成本较大;若内部变化复杂,会有很多的建造者类。
五.例子
1 //需要构建的复杂产品
2 public classMeal {3
4 privateString food;5
6 privateString drink;7
8 publicString getFood() {9 returnfood;10 }11
12 public voidsetFood(String food) {13 this.food =food;14 }15
16 publicString getDrink() {17 returndrink;18 }19
20 public voidsetDrink(String drink) {21 this.drink =drink;22 }23
24 //抽下构建者接口
25 public abstract classMealBuilder {26 protected Meal meal = newMeal();27 public abstract voidbuildFood();28
29 public abstract voidbuildDrink();30
31 public abstractMeal buildMeal();32 }33
34 //具体构建者
35 /**
36 * 套餐A37 */
38 public class MealA extendsMealBuilder {39 @Override40 public voidbuildFood() {41 meal.setFood("薯条");42 }43
44 @Override45 public voidbuildDrink() {46 meal.setDrink("可乐");47 }48
49 @Override50 publicMeal buildMeal() {51 returnmeal;52 }53 }54
55 public class MealB extendsMealBuilder {56 @Override57 public voidbuildFood() {58 meal.setFood("汉堡");59 }60
61 @Override62 public voidbuildDrink() {63 meal.setDrink("橙汁");64 }65
66 @Override67 publicMeal buildMeal() {68 returnmeal;69 }70 }
测试结果: