建造者模式

本文参考:《修炼Java开发技术:在架构中体验设计模式和算法之美   于广编著》。

 

  在软件开发过程中,当遇到一个“复杂的对象”,该对象由好多部分组成,各个部分的组合比较稳定或者有一定的依赖次序,但各个部分自身却会经常面临着变化时,冷䲽隔离出“复杂对象的各个部分”的变化,从而保持系统中的“稳定构件算法”,这就是建造者模式的任务。

  建造者模式是将复杂的内部创建封装在内部,对于外部调用的人来说,只需要使传入建造者和建造工具,关于内部是如何建造成成品的,调用者无需关心。

 

适用场景:

1、需要生成的产品有复杂的内部结构。

2、需要生成的产品对象的属性互相依赖。

3、在对象的创建过程中会使用到其他的对象。

 

建造者模式的结构:

Director<>--(聚合)--> Builder <|--(泛化)-- ConcreteBuilder<--(依赖)--Product

 

Builer(抽象建造者):它为创建一个产品product对象的各个部件指定抽象接口,在该接口中一般声明两类方法,一类buildPartX(),它们用于创建复杂对象的各个部件;另一类方法时getResult(),它们用于返回复杂对象。Builder既可以是抽象类,也可以是接口。

ConcreteBuilder(具体建造者):它实现了Builder接口,实现各个部件的具体构造和装配方法,定义并明确它所创建的复杂对象,也可以提供一个方法返回创建好的复杂产品对象。

Product(产品角色):它是被构建的复杂对象,包含多个组成部件,集体建造者创建该产品的内部表示并定义它的装配过程。

Director(指挥者):指挥者又被称为导演类,它负责安排复杂对象的建造次序,指挥者与抽象建造者之间存在关联关系,可以在其构件方法construct()中调用建造者对象的部件构造与装配方法,完成建造复杂对象的任务。客户端一般与指挥者进行交互,在客户端确定具体建造者的类型,并实例化具体建造者对象,然后通过指挥者类的构造函数或者setter方法将该对象传入指挥者类中。

 

举例说明:

加入要通过一个汽车工厂组装一辆汽车。其中汽车由车头、车身和车位3部分组成,它的基本组装步骤如下:1、组装车头;2、组装车身;3、组装车尾。

不管要创建吉普车、卡车、公交车,它们都可以各自重新定义车头、车身和车尾的组装方法。而通过这些被重新定义的方法和相同的组装步骤,就可以组装具有不同属性的各类汽车。

由此可以知道,上述场景满足Builder模式的应用场景所提到的如下条件:

1、对象的创建:我的需要创建对象;

2、创建的是一个复合对象:汽车的头部,车身,尾部等复合对象;

3、关注对象创建的各部分的创建过程:吉普车和卡车等对车头、车身、车尾的组装方法不尽相同。

设计类如下:

CarDirector:汽车组装操作的封装类;

CarBuilder:汽车组装抽象类;

JeepBuilder:吉普车组装类,继承CarBuilder类;

Car:汽车类,包括车头、车身、车尾等属性。由JeepBuilder等创建。

package org.dyb.design.builder;

public class Car {
    private String head;//车头
    private String body;//车身
    private String tail;//车尾
    public String getHead() {
        return head;
    }
    public void setHead(String head) {
        this.head = head;
    }
    public String getBody() {
        return body;
    }
    public void setBody(String body) {
        this.body = body;
    }
    public String getTail() {
        return tail;
    }
    public void setTail(String tail) {
        this.tail = tail;
    }
    @Override
    public String toString(){
        return head+"-"+body+"-"+tail;
    }
}

 

package org.dyb.design.builder;

public abstract class CarBuilder {
    public abstract void makeHead();
    public abstract void makeBody();
    public abstract void makeTail();
    public abstract Car getCar();//得到汽车对象
}

 

package org.dyb.design.builder;

public class JeepBuilder extends CarBuilder {
    private Car car = new Car();
    @Override
    public void makeHead() {
        car.setHead("jeep head");
    }

    @Override
    public void makeBody() {
        car.setBody("jeep body");
    }

    @Override
    public void makeTail() {
        car.setTail("jeep tail");
    }

    @Override
    public Car getCar() {
        return car;
    }

}

 

package org.dyb.design.builder;

public class CarDirector {
 
    public void makeCar(CarBuilder builder){
        builder.makeHead();//组装车头
        builder.makeBody();//组装车身
        builder.makeTail();//组装车尾
    }
}

 测试:

package org.dyb.design.builder;

import org.junit.Test;

public class TestBuilder {

    @Test
    public void test(){
        CarDirector carDirector = new CarDirector();
        CarBuilder b = new JeepBuilder();
        carDirector.makeCar(b);
        Car car = b.getCar();
        System.out.println(car.toString());
    }
}

 结果:jeep head-jeep body-jeep tail

 

建造者模式与工厂模式的区别:

关注点不同。工厂模式只关心你要的是什么,而不关心东西的具体细节是什么。建造者模式则关心的是这个东西的具体细节的创建。

 

建造者模式的总结:

优点:

1、封装性。

  可以使客户端不必知道产品内部组成的细节。

2、建造者独立,容易扩展。

3、便于控制细节风险。

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

 

使用场景:

1、相同的方法,不同的执行顺序,产生不同的事件结果。

2、多个部件或者零件,都可以装配到一个对象中,但是产生的运行结果又不相同。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值