跟屌丝大哥学习设计模式--生成器模式(Builder)

简介:

Builder 模式的重心在于分离构建算法和具体的构造实现,从而使构建算法可以重用。

Builder 模式的构成分为两部分:一部分是Builder接口,定义了如何构建各个部件,并装配到产品中去;另一部分是Director,定义如何来构建产品,Director 负责整体的构建算法,而且通常是分步来执行的。

注:这里的构建算法是指:通过什么样的方式来组装产品;构建产品指的是:构建一个复杂对象。

Builder 模式就是将构建产品部件组装产品的过程分开,即实现了产品部件组装产品过程的解耦,可以使得组装产品过程得到复用



public class ExportHeaderModel {

         private String depId;
         private String exportDate;
         省略getter 和 setter
 } 

public class ExportDataModel {

       private String productId;
       private double price;
       private double amount;
       省略getter 和 setter
}

public class ExportFooterModel {

      private String exportUser;

      省略getter 和 setter

 }
/**
 * 生成器接口,定义一个输出文件对象所需的各个部件的操作
 *
 */
public interface Builder {

          /**
           * 构建输出文件的Header部分
           * @param ehm 文件头的内容
           */
         public void buildHeader(ExportHeaderModel ehm);
         /**
          * 构建输出文件的Body部分
          * @param mapData 要输出文件的数据内容
          */
        public void buildBody(Map<String, Collection<ExportDataModel>> mapData);
        /**
         * 构建要输出文件的Footer部分
         * @param efm 文件尾的内容
         */
        public void buildFooter(ExportFooterModel efm);
}

/**
 * 实现导出数据到文本文件的生成器
 *
 */

public class TxtBuilder implements Builder {

          /**
           * 用来记录构建文件的内容,相当于产品
           */
          private StringBuffer buffer = new StringBuffer();
 
          @Override
          public void buildHeader(ExportHeaderModel ehm) {
                    buffer.append(ehm.getDepId() + "," + ehm.getExportDate() + "\n");
           }

          @Override
          public void buildBody(Map<String, Collection<ExportDataModel>> mapData) {
                   for(String tblName : mapData.keySet()){
                           buffer.append(tblName + "\n");
                           for(ExportDataModel edm : mapData.get(tblName)){
                                    buffer.append(edm.getProductId() + "," + edm.getPrice() + "," + edm.getAmount() + "\n");
                            }
                   }
         }

   @Override
         public void buildFooter(ExportFooterModel efm) {
                    buffer.append(efm.getExportUser());
         }

         public StringBuffer getResult(){
                   return buffer;
         }
}

public class XmlBuilder implements Builder {

          private StringBuffer buffer = new StringBuffer();
 

  @Override
           public void  buildHeader(ExportHeaderModel ehm) {
                    buffer.append("<?xml version='1.0' encoding='gb2312' ?>\n");
                    buffer.append("<Report>\n");
                    buffer.append("  <Header>\n");
                    buffer.append("    <DepId>" + ehm.getDepId() + "</DepId>\n");
                    buffer.append("    <ExportDate>" + ehm.getExportDate() + "</ExportDate>\n");
                    buffer.append("  </Header>\n");
         }
     @Override
          public void  buildBody(Map<String, Collection<ExportDataModel>> mapData) {
                  buffer.append("  <Body>\n");
                   for (String tblName : mapData.keySet()){
                          buffer.append("    <Datas TableName=\"" + tblName + "\">\n");
                           for (ExportDataModel edm : mapData.get(tblName)){
                                     buffer.append("      <Data>\n");
                                     buffer.append("        <ProductId>" + edm.getProductId() + "</ProductId>\n");
                                     buffer.append("        <Price>" + edm.getPrice() + "</Price>\n");
                                     buffer.append("        <Amount>" + edm.getAmount() + "</Amount>\n");
                                     buffer.append("      </Data>\n");
                          }
                         buffer.append("    </Datas>\n");
                }
                buffer.append("  </Body>\n");
        }

      @Override
         public void buildFooter(ExportFooterModel efm) {
                   buffer.append("  <Footer>\n");
                   buffer.append("    <ExportUser>" + efm.getExportUser() + "</ExportUser>\n");
                   buffer.append("  </Footer>\n");
                   buffer.append("</Report>\n");
         }

         public StringBuffer getResult(){
                  return buffer;
         }
}

/**
 * 指导者,指导使用生成器的接口来构建输出的文件对象
 *

/**
 * 指导者,指导使用生成器的接口来构建输出的文件对象
 *
 */
public class Director {

          /**
           * 持有当前需要使用的生成器对象
           */
         private Builder builder;
 
         public Director(Builder builder){
                   this.builder = builder;
         }
         /**
          * 指导生成器构建最终的输出文件的对象
          * @param ehm 文件头的内容
          * @param mapData 数据的内容
          * @param efm 文件尾的内容
          */
        public void construct(ExportHeaderModel ehm, Map<String, Collection<ExportDataModel>> mapData,
                ExportFooterModel efm){
                     //构建Header
                     builder.buildHeader(ehm);
                     //构建Body
                     builder.buildBody(mapData);
                    //构建Footer
                     builder.buildFooter(efm);
        }
}

public class Client {

          public static void main(String[] args) {
                  ExportHeaderModel ehm = new ExportHeaderModel();
                  ehm.setDepId("一分公司");
                  ehm.setExportDate("2011-06-12");
  
                  Map<String, Collection<ExportDataModel>> mapData = new HashMap<String, Collection<ExportDataModel>>();
                  Collection<ExportDataModel> coll = new ArrayList<ExportDataModel>();
  
                  ExportDataModel edml = new ExportDataModel();
                  edml.setAmount(80);
                  edml.setProductId("产品001号");
                  edml.setPrice(100);
                  coll.add(edml);
  
                 ExportDataModel edm2 = new ExportDataModel();
                 edm2.setAmount(60);
                 edm2.setProductId("产品002号");
                 edm2.setPrice(120);
                 coll.add(edm2);
  
                mapData.put("销售记录表", coll);
  
                ExportFooterModel efm = new ExportFooterModel();
                efm.setExportUser("张三");
  
                TxtBuilder txtBuilder = new TxtBuilder();
                Director director = new Director(txtBuilder);
                director.construct(ehm, mapData, efm);
                System.out.println("输出到文本文件的内容:\n" + txtBuilder.getResult());
  
                XmlBuilder xmlBuilder = new XmlBuilder();
               Director director2 = new Director(xmlBuilder);
               director2.construct(ehm, mapData, efm);
               System.out.println("输出到Xml文件的内容:\n" + xmlBuilder.getResult());
        }

}

● 使用生成器模式创建复杂对象:

① 由于使用Builder 模式来创建某个对象,因此就没有必要再定义一个Builder接口,直接提供一个具体的构建器类就可以了。

② 对于创建一个复杂的对象,可能会有很多种不同的选择和步骤,干脆去掉“Director”,把Director的功能和Client 的功能合并起来,也就是说这个时候,Client 相当于指导者,它来指导构建器类去构建需要的复杂对象。

public class ConcreteBuilder {

        private String contractId;
        private String personName;
        private String companyName;
        private long beginDate;
        private long endDate;
        private String otherData;
 
       /**
         * 构造方法 传入必填数据
         * @param contractId 保险合同号
         * @param beginDate 保险开始生效的日期
         * @param endDate 保险失效的日期
         */
       public ConcreteBuilder(String contractId, long beginDate, long endDate){
                this.contractId = contractId;
                this.beginDate = beginDate;
                this.endDate = endDate;
        }
        /**
          * 选填数据,被保险人
          * @param personName 被保险人名
          * @return 构建对象
          */
        public ConcreteBuilder setPersonName(String personName){
                  this.personName = personName;
                  return this;
        }
        /**
         * 选填数据,被保险公司
         * @param companyName 被保险公司名
         * @return 构建对象
         */
       public ConcreteBuilder setCompanyName(String companyName){
                 this.companyName = companyName;
                 return this;
        }
        /**
         * 选填数据,其它数据
         * @param otherData 其它数据
         * @return 构建对象
         */
        public ConcreteBuilder setOtherData(String otherData){
                this.otherData = otherData;
                return this;
        }
 
        public InsuranceContract build(){
               if(contractId == null || contractId.trim().length() == 0){
                        throw new IllegalArgumentException("合同编号不能空!");
                }
  
                boolean signPerson = (personName != null && personName.trim().length() > 0);
                boolean signCompany = (companyName != null && companyName.trim().length() > 0);
  
                if(!(signPerson ^ signCompany)){
                            throw new IllegalArgumentException("一份保险不能没有签订对象,且不能同时与人和公司签订!");
                }
  
               if(beginDate <= 0){
                         throw new IllegalArgumentException("合同必须有保险开始生效的日期!");
                }
  
               if(endDate <= 0){
                        throw new IllegalArgumentException("合同必须有保险失效的日期!");
               }
  
               if(endDate <= beginDate){
                        throw new IllegalArgumentException("保险失效日期必须大于生效日期!");
               }
  
                return new InsuranceContract(this);
         }
 
          public String getContractId() {
                    return contractId;
          }
          public String getPersonName() {
                    return personName;
          }
          public String getCompanyName() {
                    return companyName;
          }
          public long getBeginDate() {
                     return beginDate;
          }
          public long getEndDate() {
                    return endDate;
          }
          public String getOtherData() {
                    return otherData;
          }
}

/**
 * 保险合同对象
 * @author joe
 *
 */
public class InsuranceContract {

        /**
         * 保险合同编号
         */
       private String constractId;
       /**
        * 被保险的人
        */
       private String personName;
       /**
        * 被保险的公司
        */
      private String companyName;
      /**
       * 保险开始生效的日期
      */
      private long beginDate;
      /**
       * 保险失效的日期
       */
      private long endDate;
      /**
       * 其它数据
       */
      private String otherData;
      /**
       * 构造方法,访问级别是同包能访问
       * @param builder
       */
      InsuranceContract(ConcreteBuilder builder){
             this.constractId = builder.getContractId();
             this.personName = builder.getPersonName();
             this.companyName = builder.getCompanyName();
             this.beginDate = builder.getBeginDate();
             this.endDate = builder.getEndDate();
             this.otherData = builder.getOtherData();
       }
 
      public void someOperation(){
              System.out.println("Now in Insurance Contract someOperation == " + this.constractId);
       }

}

public class Client {

          public static void main(String[] args) {
                    ConcreteBuilder builder = new ConcreteBuilder("001", 82345L, 67890L);
                    InsuranceContract contract = builder.setPersonName("张三").setOtherData("test").build();
                    contract.someOperation();
          }
}








====================================分割线================================



最新内容请见作者的GitHub页:http://qaseven.github.io/

目录
相关文章
|
8天前
|
设计模式 SQL 算法
设计模式了解哪些,模版模式
设计模式了解哪些,模版模式
18 0
|
27天前
|
设计模式 Java uml
C++设计模式之 依赖注入模式探索
C++设计模式之 依赖注入模式探索
37 0
|
3月前
|
设计模式 存储 算法
Java 设计模式最佳实践:三、行为模式
Java 设计模式最佳实践:三、行为模式
22 0
|
2月前
|
设计模式 前端开发 JavaScript
观察者模式 vs 发布-订阅模式:两种设计模式的对决!
欢迎来到前端入门之旅!这个专栏是为那些对Web开发感兴趣、刚刚开始学习前端的读者们打造的。无论你是初学者还是有一些基础的开发者,我们都会在这里为你提供一个系统而又亲切的学习平台。我们以问答形式更新,为大家呈现精选的前端知识点和最佳实践。通过深入浅出的解释概念,并提供实际案例和练习,让你逐步建立起一个扎实的基础。无论是HTML、CSS、JavaScript还是最新的前端框架和工具,我们都将为你提供丰富的内容和实用技巧,帮助你更好地理解并运用前端开发中的各种技术。
|
4天前
|
设计模式 Java 数据库
小谈设计模式(2)—简单工厂模式
小谈设计模式(2)—简单工厂模式
|
3月前
|
设计模式 监控 安全
多线程设计模式【多线程上下文设计模式、Guarded Suspension 设计模式、 Latch 设计模式】(二)-全面详解(学习总结---从入门到深化)
多线程设计模式【多线程上下文设计模式、Guarded Suspension 设计模式、 Latch 设计模式】(二)-全面详解(学习总结---从入门到深化)
61 0
|
4天前
|
设计模式 Java
小谈设计模式(9)—工厂方法模式
小谈设计模式(9)—工厂方法模式
|
1月前
|
设计模式 编译器
解析器模式--设计模式
解析器模式--设计模式
17 0
|
1月前
|
设计模式 算法
构建器模式--设计模式
构建器模式--设计模式
16 0
|
1月前
|
设计模式
【设计模式】中介模式
【设计模式】中介模式