Chinaunix首页 | 论坛 | 博客
  • 博客访问: 6333160
  • 博文数量: 2759
  • 博客积分: 1021
  • 博客等级: 中士
  • 技术积分: 4091
  • 用 户 组: 普通用户
  • 注册时间: 2012-03-11 14:14
文章分类

全部博文(2759)

文章存档

2019年(1)

2017年(84)

2016年(196)

2015年(204)

2014年(636)

2013年(1176)

2012年(463)

分类: Java

2014-02-15 02:29:08

建造者模式也叫生成器模式,23种设计模式之一,属于创建类模式。
在描述建造者模式之前,大家首先要明白一个道理就是任何设计模式的产生都是为了解决某一类问题,更准确的说是为了应对某一种变化,设计模式就是封装变化,把变化的部分和不变的部分隔离开。设计模式的好处就是运用面向对象的思想让我们能够更简单方便的应对外界需求的变化。比如工厂模式的产生是由于我们要根据外界的需求生产出不同类型的产品,这一变化就是外界的需求是不确定的,工厂模式就是为了应对这种变化而产生。那建造者模式是为了应对哪一种变化呢?
首先还是看下GOF对建造者模式的定义:将一个复杂对象的构建与其表示相分离,使得同样的构建过程可以创建出不同的表示。GOF的语言一般都比较言简意赅,但也有点晦涩难读。
看到有网友用KFC的例子来描述建造者模式,我觉得还是比较贴近的,我们也借用这个例子描述一下:
假设KFC推出两种套餐:奥尔良鸡腿堡套餐和香辣鸡腿堡套餐,奥尔良套餐包括:一个奥尔良鸡腿堡、一个炸鸡翅、一杯雪碧。鸡腿堡套餐包括:一个香辣鸡腿堡、一份薯条、一杯可乐。每份套餐都是:主食、副食、饮料。
KFC服务员要根据顾客的要求来提供套餐,那这个需求里面什么是固定的,什么是变化的呢?很明显顾客都是要的套餐,顾客的目的是一样的。 套餐里面都是主食、副食、饮料,这也是固定的。至于主食是什么、副食是什么、饮料是什么这个是变化的。
在软件系统中,有时候面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成,由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将他们组合在一起的算法却相对稳定。当遇到这种场景时,你就可以考虑能否用建造者模式实现了。
下面我们用一个例子描述下建造者模式的代码实现过程。


1、第一个文件House.java

点击(此处)折叠或打开

  1. package com.myjava.interview.enchiridion.designpattern.builderpattern.model;

  2. public class House {

  3.     private String door;
  4.     
  5.     private String floor;
  6.     
  7.     private String wall;
  8.     
  9.     private String windows;

  10.     public String getDoor() {
  11.         return door;
  12.     }

  13.     public void setDoor(String door) {
  14.         this.door = door;
  15.     }

  16.     public String getFloor() {
  17.         return floor;
  18.     }

  19.     public void setFloor(String floor) {
  20.         this.floor = floor;
  21.     }

  22.     public String getWall() {
  23.         return wall;
  24.     }

  25.     public void setWall(String wall) {
  26.         this.wall = wall;
  27.     }

  28.     public String getWindows() {
  29.         return windows;
  30.     }

  31.     public void setWindows(String windows) {
  32.         this.windows = windows;
  33.     }
  34. }
2、第二个文件ChinaStyleHouse.java

点击(此处)折叠或打开

  1. package com.myjava.interview.enchiridion.designpattern.builderpattern.model;

  2. public class ChinaStyleHouse extends House {
  3.     //do something here
  4. }

3、第三个文件BritishStyleHouse.java

点击(此处)折叠或打开

  1. package com.myjava.interview.enchiridion.designpattern.builderpattern.model;

  2. public class BritishStyleHouse extends House {
  3.     //do something here
  4. }

4、第四个文件IHouseBuilder.java
 

点击(此处)折叠或打开

  1. package com.myjava.interview.enchiridion.designpattern.builderpattern.service;

  2. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.House;

  3. public interface IHouseBuilder {
  4.     
  5.     /**
  6.      * 建造墙壁
  7.      */
  8.     public void buildWall();
  9.     
  10.     /**
  11.      * 建造房门
  12.      */
  13.     public void buildDoor();
  14.     
  15.     /**
  16.      * 建造地板
  17.      */
  18.     public void buildFloor();
  19.     
  20.     /**
  21.      * 建造窗户
  22.      * @return
  23.      */
  24.     public void buildWindows();
  25.     
  26.     /**
  27.      * 返回创造完成的对象
  28.      * @return
  29.      */
  30.     public House getHouse();
  31. }

5、第五个文件ChinaStyleHouseBuilderImpl.java
 

点击(此处)折叠或打开

  1. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.ChinaStyleHouse;
  2. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.House;
  3. import com.myjava.interview.enchiridion.designpattern.builderpattern.service.IHouseBuilder;

  4. public class ChinaStyleHouseBuilderImpl implements IHouseBuilder{

  5.     public House house = new ChinaStyleHouse();
  6.     
  7.     @Override
  8.     public void buildDoor() {
  9.         house.setDoor("中式风格房门");
  10.     }

  11.     @Override
  12.     public void buildFloor() {
  13.         house.setFloor("中式风格地板");
  14.     }

  15.     @Override
  16.     public void buildWall() {
  17.         house.setWall("中式风格墙壁");
  18.     }

  19.     @Override
  20.     public void buildWindows() {
  21.         house.setWindows("中式风格门窗");
  22.     }

  23.     @Override
  24.     public House getHouse() {
  25.         return house;
  26.     }

  27. }
6、第六个文件BritishStyleHouseBuilderImpl.java

点击(此处)折叠或打开

  1. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.BritishStyleHouse;
  2. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.House;
  3. import com.myjava.interview.enchiridion.designpattern.builderpattern.service.IHouseBuilder;

  4. public class BritishStyleHouseBuilderImpl implements IHouseBuilder {

  5.     public House house = new BritishStyleHouse();
  6.     
  7.     @Override
  8.     public void buildDoor() {
  9.         house.setDoor("英式风格门");
  10.     }

  11.     @Override
  12.     public void buildFloor() {
  13.         house.setFloor("英式风格地板");
  14.     }

  15.     @Override
  16.     public void buildWall() {
  17.         house.setWall("英式风格地板");
  18.     }

  19.     @Override
  20.     public void buildWindows() {
  21.         house.setWindows("英式风格窗户");
  22.     }

  23.     @Override
  24.     public House getHouse() {
  25.         return house;
  26.     }

  27. }

7、第七个文件HouseDirector.java

点击(此处)折叠或打开

  1. package com.myjava.interview.enchiridion.designpattern.builderpattern.director;

  2. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.House;
  3. import com.myjava.interview.enchiridion.designpattern.builderpattern.service.IHouseBuilder;

  4. public class HouseDirector {

  5.     public House buildHouse(IHouseBuilder builder){
  6.         /**
  7.          * 建造房门
  8.          */
  9.         builder.buildDoor();
  10.         /**
  11.          * 建造地板
  12.          */
  13.         builder.buildFloor();
  14.         /**
  15.          * 建造墙壁
  16.          */
  17.         builder.buildWall();
  18.         /**
  19.          * 建造窗户
  20.          */
  21.         builder.buildWindows();
  22.         
  23.         return builder.getHouse();
  24.     }
  25. }
8、第八个文件TestMain.java

点击(此处)折叠或打开

  1. import com.myjava.interview.enchiridion.designpattern.builderpattern.director.HouseDirector;
  2. import com.myjava.interview.enchiridion.designpattern.builderpattern.model.House;
  3. import com.myjava.interview.enchiridion.designpattern.builderpattern.service.IHouseBuilder;
  4. import com.myjava.interview.enchiridion.designpattern.builderpattern.service.impl.BritishStyleHouseBuilderImpl;
  5. import com.myjava.interview.enchiridion.designpattern.builderpattern.service.impl.extend.ChinaStyleAHouseBuilderImpl;

  6. public class TestClient {

  7.     public static void main(String [] args){
  8.         IHouseBuilder cbuilder = new ChinaStyleHouseBuilderImpl();
  9.         House chouse = new HouseDirector().buildHouse(cbuilder);
  10.         System.out.println("------------------------------");
  11.         System.out.println("房门属性:" + chouse.getDoor());
  12.         System.out.println("墙壁属性:" + chouse.getWall());
  13.         System.out.println("天花板属性:" + chouse.getFloor());
  14.         System.out.println("窗户属性:" + chouse.getWindows());
  15.         System.out.println("------------------------------");
  16.         IHouseBuilder ebuilder = new BritishStyleHouseBuilderImpl();
  17.         House ehouse = new HouseDirector().buildHouse(ebuilder);
  18.         System.out.println("房门属性:" + ehouse.getDoor());
  19.         System.out.println("墙壁属性:" + ehouse.getWall());
  20.         System.out.println("天花板属性:" + ehouse.getFloor());
  21.         System.out.println("窗户属性:" + ehouse.getWindows());
  22.         System.out.println("------------------------------");
  23.     }
  24.     
  25. }
如上功能代码中House对象就如同KFC例子中的套餐,它是一个抽象的概念,而ChinaStyleHouse对象和BritishStyleHouse对象是具体的两种产品如同KFC例子中的奥尔良套餐和鸡腿堡套餐。
IHouseBuilder是一个抽象的建造者对象,而ChinaStyleHouseBuilderImpl和BritishStyleHouseBuilderImpl就是两种产品的实际建造者,它们对应了KFC例子中制造两种套餐的工人,这部分功能的代码相对来说是经常变化的,而HouseDiretor中的逻辑相对来说是固定不变的。

 9、建造者模式的优点

(1使用建造者模式可以使客户端不必知道产品内部组成的细节。

(2体的建造者类之间是相互独立的,对系统的扩展非常有利。

(3于具体的建造者是独立的,因此可以对建造过程逐步细化,而不对其他的模块产生任何影响。

阅读(1359) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~