设计模式系列之「建造者模式」

掘金地址:https://juejin.im/post/5a30d5af6fb9a0450167f2d7

欢迎收看俗到掉渣的《小Y讲堂》节目,大家好,我是小Y,一个集性感毛发与才华于一身的程序猿!近日收到《魂斗罗.归来》中的肌肉男比尔·雷泽的投诉,说要投诉小Y最近冷落他,太久没有让他上节目show muscle。没办法,为了满足这个闷骚的老男人,小Y把这次的主题设置为如果比尔是程序员,会怎么用建造者模式来实现关卡武器装配。oh,my God,很难想象战斗狂人叼着雪茄在死命敲代码的情形(一阵恶寒啊),得赶紧来幅小Y牌“止吐”图来镇镇。

一、初出茅庐的比尔·雷泽

比尔最近迷上了编程,刚学到点三脚猫功夫就吵着要写段代码为自己代言,要把自己不同的形象展现出来,比尔写了以下代码:

①角色的基配

public class Character {
    //赤裸裸的比尔雷泽,出战前需要做充能、装备主武器以及副武器
    private Energy energy;
    private MainWeapon mainWeapon;
    private ViceWeapon viceWeapon;

    public Character(Energy energy, MainWeapon mainWeapon, ViceWeapon viceWeapon) {
        this.energy = energy;
        this.mainWeapon = mainWeapon;
        this.viceWeapon = viceWeapon;
    }

    @Override
    public String toString() {
        return energy.getEnergy()+mainWeapon.getMainWeapon()+viceWeapon.getViceWeapon();
    }
}

②出战前基配类型(充能、选择主武器以及副武器)

//充能
public abstract class Energy {
    public abstract String getEnergy();
}

//主武器
public abstract class MainWeapon {
    public abstract String getMainWeapon();
}

//副武器
public abstract class ViceWeapon {
    public abstract String getViceWeapon();
}

③1VS1的武器装配

//1VS1的充能
public class OneVsOneEnergy extends Energy{
    @Override
    public String getEnergy() {
        return "充能完成。";
    }
}

//1VS1的主武器装配
public class OneVsOneMainWeapon extends MainWeapon{

    @Override
    public String getMainWeapon() {
        return "主武器:黄金加特林。";
    }
}

//1VS1的副武器装配
public class OneVsOneViceWeapon extends ViceWeapon{
    @Override
    public String getViceWeapon() {
        return "副武器:集速手雷。";
    }
}

④3VS3的武器装配

//3VS3的充能
public class ThreeVsThreeEnergy extends Energy{
    @Override
    public String getEnergy() {
        return "充能完成。";
    }
}

//3VS3的主武器装配
public class ThreeVsThreeMainWeapon extends MainWeapon{

    @Override
    public String getMainWeapon() {
        return "主武器:突击步枪。";
    }
}

//3VS3的副武器装配
public class ThreeVsThreeViceWeapon extends ViceWeapon{
    @Override
    public String getViceWeapon() {
        return "副武器:等离子喷射器。";
    }
}

⑤Client实现

public class Client {
    public static void main(String[] args){
        //1VS1下的比尔
        Character OneVsOneOfBill=new Character(new OneVsOneEnergy(),new OneVsOneMainWeapon(),new OneVsOneViceWeapon());
        System.out.println(OneVsOneOfBill);
        //3VS3下的比尔
        Character threeVThreeOfBill=new Character(new ThreeVsThreeEnergy(),new ThreeVsThreeMainWeapon(),new ThreeVsThreeViceWeapon());
        System.out.println(threeVThreeOfBill);

    }
}

输出结果

充能完成。主武器:黄金加特林。副武器:集速手雷。 
充能完成。主武器:突击步枪。副武器:等离子喷射器。 

对于一个刚学习编程的比尔来说,撇开设计模式来说,能够写出这样的代码,小Y都是佩服得不要不要的了,但是为了唬住这个没长全毛的比尔,小Y毅然搬出了设计模式,对比尔进行了义正言辞的批评教育:

  • 随着等级的越来越高,面对的关卡种类就会越来越多,这也意味着不同的关卡的主副武器的搭配的种类就会越来越多,况且这个例子只是一个简单的传参,如果传参复杂点还按照这种传参的方式进行会很容易搞混,出现搭配不对的情况。
  • 产品的内部组成暴露给客户端,封装性差。

为了防止比尔反驳,小Y立马抛出建造者模式

二、基本概念

1.定义

将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。

2.理解

就是把一个产品(对象)表示(展示)和构建(创建)过程分离开来,这样产品的构建流程相同却可以有不同的产品表示。

3.为何使用建造者模式
  • 是为了将构建复杂对象的过程和它的部件解耦。
  • 建造者模式的封装性很好。使用建造者模式可以有效的封装变化。
4.应用场景
  • 同一个创建过程需要有不同的内部表象的产品对象。
  • 创建复杂对象的算法独立于组成对象的部件。
5.角色介绍

  • Director导演类
    负责调用适当的建造者来组建产品,导演类一般不与产品类发生依赖关系,与导演类直接交互的是建造者类。

  • Builder抽象建造者
    规范产品的组建,一般是由子类实现。

  • ConcreteBuilder具体建造者
    实现抽象类的所有未实现的方法,具体来说一般是两项任务:组建产品;返回组建好的产品。。

  • Product产品类
    由一系列部件组成,一般是一个较为复杂的对象,也就是说创建对象的过程比较复杂,一般会有比较多的代码量。

  • 建造模式分成两个很重要的部分:
    一个部分是Builder接口,这里是定义了如何构建各个部件,也就是知道每个部件功能如何实现,以及如何装配这些部件到产品中去;另外一个部分是Director,Director是知道如何组合来构建产品,也就是说Director负责整体的构建算法,而且通常是分步骤地来执行。
     

三、案列实现

经过小Y孜孜不倦的教诲,比尔·雷泽总算是领悟了建造者模式的精髓,决定了重新修改了修改一下上面的代码,经过整理得到:

1.UML清单

2.代码实现

修改后角色的基配

public class Character {
    //赤裸裸的比尔雷泽,出战前需要做充能、装备主武器以及副武器
    private Energy energy;
    private MainWeapon mainWeapon;
    private ViceWeapon viceWeapon;

    public Energy getEnergy() {
        return energy;
    }

    public void setEnergy(Energy energy) {
        this.energy = energy;
    }

    public MainWeapon getMainWeapon() {
        return mainWeapon;
    }

    public void setMainWeapon(MainWeapon mainWeapon) {
        this.mainWeapon = mainWeapon;
    }

    public ViceWeapon getViceWeapon() {
        return viceWeapon;
    }

    public void setViceWeapon(ViceWeapon viceWeapon) {
        this.viceWeapon = viceWeapon;
    }

    @Override
    public String toString() {
        return energy.getEnergy()+mainWeapon.getMainWeapon()+viceWeapon.getViceWeapon();
    }
}

②角色建造抽象类

public interface CharacterBuilder {
    void makeEnergy();
    void makeMainWeapon();
    void makeViceWeapon();

    public Character build();
}

③1VS1具体建造者

public class OneVsOneBulider implements CharacterBuilder {

    private Character character;

    public OneVsOneBulider() {
        this.character = new Character();
    }

    @Override
    public void makeEnergy() {
        character.setEnergy(new OneVsOneEnergy());
    }

    @Override
    public void makeMainWeapon() {
        character.setMainWeapon(new OneVsOneMainWeapon());
    }

    @Override
    public void makeViceWeapon() {
        character.setViceWeapon(new OneVsOneViceWeapon());
    }

    @Override
    public Character build() {
        return character;
    }
}

④3VS3具体建造者

public class ThreeVsThreeBulider implements CharacterBuilder {

    private Character character;

    public ThreeVsThreeBulider() {
        character = new Character();
    }

    @Override
    public void makeEnergy() {
        character.setEnergy(new ThreeVsThreeEnergy());
    }

    @Override
    public void makeMainWeapon() {
        character.setMainWeapon(new ThreeVsThreeMainWeapon());
    }

    @Override
    public void makeViceWeapon() {
        character.setViceWeapon(new ThreeVsThreeViceWeapon());
    }

    @Override
    public Character build() {
    return character;
    }
}

⑤Director导演类

public class CharacterDirector {

    private CharacterBuilder characterBuilder;

    public CharacterDirector(CharacterBuilder characterBuilder) {
        this.characterBuilder = characterBuilder;
    }

    public Character createCharacter(){
        characterBuilder.makeEnergy();
        characterBuilder.makeMainWeapon();
        characterBuilder.makeViceWeapon();
        return characterBuilder.build();
    }
}

⑥Client实现

public class Client {

    public static void main(String[] args){
        //1VS1下的比尔
        CharacterBuilder oneVsOneBulider=new OneVsOneBulider();
        CharacterDirector characterDirector=new CharacterDirector(oneVsOneBulider);
        System.out.println(characterDirector.createCharacter());
        //3VS3下的比尔
        CharacterBuilder threeVsThreeBulider=new ThreeVsThreeBulider();
        characterDirector=new CharacterDirector(threeVsThreeBulider);
        System.out.println(characterDirector.createCharacter());

    }
}

输出结果

充能完成。主武器:黄金加特林。副武器:集速手雷。 
充能完成。主武器:突击步枪。副武器:等离子喷射器

四、优缺点

1.优点

  • 封装性,使用建造者模式可以使客户端不必知道产品内部组成的细节,如例子中我们就不需要关心每一个具体的模型内部是如何实现。

  • 建造者独立,容易扩展。OneVsOneBulider和ThreeVsThreeBulider是相互独立的,对系统的扩展非常有利。

  • 便于控制细节风险。由于具体的建造者是独立的,因此可以对建造过程逐步细化,而不对其他的模块产生任何影响。

  • 增加新的具体建造者无须修改原有类库的代码,指挥者类针对抽象建造者类编程,系统扩展方便,符合“开闭原则”。

2.缺点

  • 建造者模式所创建的产品一般具有较多的共同点,其组成部分相似,如果产品之间的差异性很大,则不适合使用建造者模式,因此其使用范围受到一定的限制。

  • 如果产品的内部变化复杂,可能会导致需要定义很多具体建造者类来实现这种变化,导致系统变得很庞大。

五、总结

建造者模式关注的是零件类型和装配工艺(顺序),这是它与工厂方法模式最大不同的地方,虽然同为创建类模式,但是注重点不同。下一篇就是工厂方法模式,有兴趣的可以继续留意。

节目到了尾声了,让我们用热烈的掌声感谢重量级嘉宾比尔雷泽,好走不送。

Android技术交流吧

掘金地址:https://juejin.im/post/5a30d5af6fb9a0450167f2d7

猜你喜欢

转载自blog.csdn.net/kaoshibiguo_1/article/details/78796709