在上篇文章中,对Java Builder模式的使用体验主要是从Builder对构造器改造方面的优秀特性来说的,感觉并没有从Java Builder模式本身的功能和作用去写,因此决定再从Builder模式的作用以及在项目开发中的使用来体验下。 
    Builder 模式,即建造者模式,顾名思义,这个模式可能更多的使用在产品的组装中使用,具体说就是在软件产品的组件或模块组装的时候使用。 
    感觉网络上比较好的解释有: 
    建造者模式(Builder):将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。 

   适用的场景: 
1、当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时。 
2、当构造过程必须允许被构造的对象有不同的表示时。 

   首先我们来说下,建造者模式中三个重要的角色: 
   1、Product (生产者):该角色其实类似于工厂里的流水线,它可以生产各种各样的组件,但是并不知道怎样将这些组件进行组装,进而产生出不同的产品。 
   2、Builder(建造者):该角色类似流水线上的操作工人,它负责将不同的流水线上生产出的不同零件或组件,按照(领导Director)特定要求组装成特定的产品,并给Director提供一个build方法,让其随时可以取走已经组装好的产品。 
   3、Director(指挥者或者领导):该角色主要根据不同客户的需求,调用不同的Builder对象来组装出特定的产品,提供给客户使用。 
   为了便于理解,我们假设还有第4个角色: 
   4、Client(客户): 客户是产品需求的来源,但是客户不会和生产线或工厂的操作工打交道,他们只会开着车,和领导打交道。 

    好了,各个角色,我们都定义好了,我们寻找一个场景吧。就拿汽车的生产来说事吧。 
    我们假设有以下场景: 
    Dirctor是生产CarX品牌汽车公司的领导,ClientA和ClientB是分别来自欧洲和亚洲的客户。ClientA希望买到的CarX汽车能够达到更好的节能和经济性需求,ClientB希望买到的车更好的满足安全性和舒适性。Dirctor根据2个客户的需求,命令工程师设计了2套流水线,2条流水线生产的产品都一样,但是产品的性能不同: 
     A生产线(ProductionLineA)主要生产小排量的发动机和配置不高的ABS安全系统; 
     B生产线(ProductionLineB)主要生产排量较大的发动机和配置较高的ABS安全系统; 
     
     同时为2条生产线安排了不同的组装工程师BuilderA和BuilderB。 
     产品的组装命令和验收由Dirctor统一负责。 
     好了,自此,任务分配完毕,我们开始编码工作了。 
     首先,对上面问题的分析,我们可以发现,A生产线和B生产线生产的东西名称和功能是一样的,只是性能不同,因此可以对生产线可以进行抽象。 
    
Java代码  Java Builder模式 体验(二)_ide
  1. package com.icecode.demo;  
  2.   
  3. public abstract class AbstractProductionLine {  
  4.     //生产汽车发动机  
  5.     public abstract void createEngine();  
  6.     //生产汽车安全系统  
  7.     public abstract void createSecuritySystem();  
  8.     //生产汽车车身  
  9.     public abstract void createCarBody();  
  10. }  

   
    同样,我们也可以对建造工程师也进行抽象,如下: 
   
Java代码  Java Builder模式 体验(二)_ide
  1. package com.icecode.demo;  
  2.   
  3. public abstract class AbstractBuilder {  
  4.     //命令1:启动生产汽车零部件生产线命令  
  5.     public abstract void buildCarParts();  
  6.     //命令2:将命令1生产的零部件组装成完整的汽车  
  7.     public abstract AbstractProductionLine buildCar();  
  8. }  
  9.    

  好了,方案和任务都安排好了,我们开始具体实施了。 
  
Java代码  Java Builder模式 体验(二)_ide
  1. package com.icecode.demo.impl;  
  2.   
  3. import com.icecode.demo.AbstractProductLine;  
  4. /** 
  5.  * 流水线A 
  6.  * @author zhanche 
  7.  * 
  8.  */  
  9. public class ConcreteProductLineA extends AbstractProductLine {  
  10.   
  11.     @Override  
  12.     public void createEngine() {  
  13.         System.out.println("------------生产小排量发动机----------");  
  14.     }  
  15.   
  16.     @Override  
  17.     public void createSecuritySystem() {  
  18.         System.out.println("----------生产普通标准的安全系统-----------");  
  19.     }  
  20.   
  21.     @Override  
  22.     public void createCarBody() {  
  23.         System.out.println("----------生成汽车车身部件-----------");  
  24.     }  
  25.   
  26.   
  27. }  
  28.   
  29. package com.icecode.demo.impl;  
  30.   
  31. import com.icecode.demo.AbstractProductLine;  
  32. /** 
  33.  * 流水线B 
  34.  * @author zhanche 
  35.  * 
  36.  */  
  37. public class ConcreteProductLineB extends AbstractProductLine {  
  38.   
  39.     @Override  
  40.     public void createEngine() {  
  41.         System.out.println("------------生产标准排量的发动机----------");  
  42.     }  
  43.   
  44.     @Override  
  45.     public void createSecuritySystem() {  
  46.         System.out.println("----------生产高标准的安全系统-----------");  
  47.     }  
  48.   
  49.     @Override  
  50.     public void createCarBody() {  
  51.         System.out.println("----------生成汽车车身部件-----------");          
  52.     }  
  53.   
  54. }  
  55.   
  56.   
  57. package com.icecode.demo.impl;  
  58.   
  59. import com.icecode.demo.AbstractBuilder;  
  60. import com.icecode.demo.AbstractProductLine;  
  61. /** 
  62.  * 建造工人A 
  63.  * @author zhanche 
  64.  * 
  65.  */  
  66. public class ConcreteBuilderA extends AbstractBuilder {  
  67.     private AbstractProductLine productLineA = new ConcreteProductLineA();  
  68.   
  69.     @Override  
  70.     public void buildCarParts() {  
  71.         productLineA.createEngine();  
  72.         productLineA.createSecuritySystem();  
  73.         productLineA.createCarBody();  
  74.         System.out.println("-----生产线A完成一辆汽车零部件的生产-----");  
  75.     }  
  76.   
  77.     @Override  
  78.     public AbstractProductLine buildCar() {  
  79.         System.out.println("----------完成一辆汽车A的组装调试----------");  
  80.         return productLineA;  
  81.     }   
  82.   
  83.   
  84. }  
  85.   
  86. package com.icecode.demo.impl;  
  87.   
  88. import com.icecode.demo.AbstractBuilder;  
  89. import com.icecode.demo.AbstractProductLine;  
  90. /** 
  91.  * 建造工人B 
  92.  * @author zhanche 
  93.  * 
  94.  */  
  95. public class ConcreteBuilderB extends AbstractBuilder {  
  96.     private AbstractProductLine productLineB = new ConcreteProductLineB();  
  97.   
  98.     @Override  
  99.     public void buildCarParts() {  
  100.         productLineB.createEngine();  
  101.         productLineB.createSecuritySystem();  
  102.         productLineB.createCarBody();  
  103.         System.out.println("-----生产线B完成一辆汽车零部件的生产----");  
  104.     }  
  105.   
  106.     @Override  
  107.     public AbstractProductLine buildCar() {  
  108.         System.out.println("----------完成一辆汽车B的组装调试----------");  
  109.         return productLineB;  
  110.     }   
  111.   
  112. }  

   到此,我们的产品建造实施实体都安排好了,现在需要做的是实现Director(领导者)角色的功能,他负责下发开始启动生产零部件的命令和开始组装的命令。 
  
Java代码  Java Builder模式 体验(二)_ide
  1. package com.icecode.demo.director;  
  2.   
  3. import com.icecode.demo.AbstractBuilder;  
  4. import com.icecode.demo.AbstractProductLine;  
  5. import com.icecode.demo.impl.ConcreteBuilderA;  
  6. import com.icecode.demo.impl.ConcreteBuilderB;  
  7.   
  8. public class Director {  
  9.     private AbstractBuilder builderA = new ConcreteBuilderA();  
  10.     private AbstractBuilder builderB = new ConcreteBuilderB();  
  11.       
  12.     public AbstractProductLine getProductA() {  
  13.         builderA.buildCarParts();  
  14.         //还可以增加对产品质量抽检等功能  
  15.         return builderA.buildCar();  
  16.     }  
  17.       
  18.     public AbstractProductLine getProductB(){  
  19.         builderB.buildCarParts();  
  20.         //还可以增加对产品质量抽检等功能  
  21.         return builderB.buildCar();  
  22.     }  
  23. }  

    领导的任务和职责也安排好了,现在2个客户ClientA和ClientB来和Director签订合同,CarX工厂开始完成客户的任务。代码如下: 
  
Java代码  Java Builder模式 体验(二)_ide
  1. package com.icecode.demo.client;  
  2.   
  3. import com.icecode.demo.director.Director;  
  4. /** 
  5.  * 客户A 
  6.  * @author zhanche 
  7.  * 
  8.  */  
  9. public class ClientA {  
  10.     /** 
  11.      * @param args 
  12.      */  
  13.     public static void main(String[] args) {    
  14.         System.out.println("ClientA 完成与Director生产汽车A的合同");   
  15.         Director director = new Director();   
  16.         director.getProductA();   
  17.     }  
  18. }  
  19.   
  20. package com.icecode.demo.client;  
  21.   
  22. import com.icecode.demo.director.Director;  
  23. /** 
  24.  * 客户B 
  25.  * @author zhanche 
  26.  * 
  27.  */  
  28. public class ClientB {  
  29.     /** 
  30.      * @param args 
  31.      */  
  32.     public static void main(String[] args) {    
  33.            
  34.         System.out.println("ClientB 完成与Director生产汽车A的合同");   
  35.         Director director = new Director();   
  36.         director.getProductB();   
  37.     }  
  38.   
  39. }  

   运行ClientA输出如下结果: 
ClientA 完成与Director生产汽车A的合同 
------------生产小排量发动机---------- 
----------生产普通标准的安全系统----------- 
----------生成汽车车身部件----------- 
----生产线A完成一辆汽车零部件的生产------ 
----------完成一辆汽车A的组装调试---------- 


运行ClientB输出如下结果: 
ClientB 完成与Director生产汽车A的合同 
------------生产标准排量的发动机---------- 
----------生产高标准的安全系统----------- 
----------生成汽车车身部件----------- 
-----生产线B完成一辆汽车零部件的生产---- 
----------完成一辆汽车B的组装调试---------- 

    到此为止,我们的一个建造者模式的应用实例已经完成了。当然,对生产线和建造者的抽象还可以使用接口来实现,原理都是一样的。 
    有人总结说Builder模式其实综合了模板方法和工厂模式的思想,其实的确可以这么理解,只是建造者模式对工厂化的生产过程和参与者的角色,做了进一步的细化。大家可以根据不同的需求,选择不同的设计模式。