一.定义:
建造者模式(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 class Meal { 3 4 private String food; 5 6 private String drink; 7 8 public String getFood() { 9 return food; 10 } 11 12 public void setFood(String food) { 13 this.food = food; 14 } 15 16 public String getDrink() { 17 return drink; 18 } 19 20 public void setDrink(String drink) { 21 this.drink = drink; 22 } 23 24 // 抽下构建者接口 25 public abstract class MealBuilder { 26 protected Meal meal = new Meal(); 27 public abstract void buildFood(); 28 29 public abstract void buildDrink(); 30 31 public abstract Meal buildMeal(); 32 } 33 34 // 具体构建者 35 /** 36 * 套餐A 37 */ 38 public class MealA extends MealBuilder { 39 @Override 40 public void buildFood() { 41 meal.setFood("薯条"); 42 } 43 44 @Override 45 public void buildDrink() { 46 meal.setDrink("可乐"); 47 } 48 49 @Override 50 public Meal buildMeal() { 51 return meal; 52 } 53 } 54 55 public class MealB extends MealBuilder { 56 @Override 57 public void buildFood() { 58 meal.setFood("汉堡"); 59 } 60 61 @Override 62 public void buildDrink() { 63 meal.setDrink("橙汁"); 64 } 65 66 @Override 67 public Meal buildMeal() { 68 return meal; 69 } 70 }
测试结果: