建造者模式(Builder Pattern)

1.介绍

1.1定义

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

1.2主要作用

在用户不知道对象的建造过程和细节的情况下就可以直接创建复杂的对象。

  1. 用户只需要给出指定复杂对象的类型和内容;
  2. 建造者模式隐藏了复杂对象的创建过程(把内部的建造过程和细节隐藏起来),按顺序创建复杂对象

1.3解决的问题

  • 方便用户创建复杂的对象(不需要知道实现过程)
  • 代码复用性 & 封装性(将对象构建过程和细节进行封装 & 复用)

例子:造汽车 & 买汽车。

  1. 工厂(建造者模式):负责制造汽车(组装过程和细节在工厂内)
  2. 汽车购买者(用户):你只需要说出你需要的型号(对象的类型和内容),然后直接购买就可以使用了
    (不需要知道汽车是怎么组装的(车轮、车门、发动机、方向盘等等))

2.模式原理

2.1 UML类图 & 讲解

uml.png

讲解:

  1. 指挥者(Director)直接和客户(Client)进行需求沟通;
  2. 沟通后指挥者将客户创建产品的需求划分为各个部件的建造请求(Builder);
  3. 将各个部件的建造请求委派到具体的建造者(ConcreteBuilder);
  4. 各个具体建造者负责进行产品部件的构建;
  5. 最终构建成具体产品(Product)。

2.2 实例讲解

接下来用一个实例来对建造者模式进行更深一步的介绍。

a. 实例概况

  • 背景:小成希望去电脑城买一台组装的台式主机
  • 过程:
  1. 电脑城老板(Diretor)和小成(Client)进行需求沟通(买来打游戏?学习?看剧?)
  2. 了解需求后,电脑城老板将小成需要的主机划分为各个部件(Builder)的建造请求(CPU、主板blabla)
  3. 指挥装机人员(ConcreteBuilder)去构建组件;
  4. 将组件组装成小成需要的电脑(Product)

b. 使用步骤

步骤1:定义具体产品类(Product):电脑
public class Computer {
    //电脑组件的集合
    private List<String> parts = new ArrayList<String>();

    //用于将组件组装到电脑里
    public void Add(String part){
        parts.add(part);
    }

    public void show(){
        for (int i = 0;i<parts.size();i++){
            System.out.println("组件"+parts.get(i)+"装好了");
        }
        System.out.println("电脑组装完成,请验收");
    }
}
步骤2:定义组装的过程(Builder):组装电脑的过程
public abstract class Builder {
    //第一步:装CPU
    //声明为抽象方法,具体由子类实现
    public abstract void  BuildCPU();

    //第二步:装主板
    //声明为抽象方法,具体由子类实现
    public abstract void BuildMainboard();

    //第三步:装硬盘
    //声明为抽象方法,具体由子类实现
    public abstract void BuildHD();

    //返回产品的方法:获得组装好的电脑
    public abstract Computer getComputer();
}
步骤3:创建具体的建造者(ConcreteBuilder):装机人员1
//装机人员1
public class ConcreteBuilder1 extends Builder{
    //创建产品实例
    Computer computer = new Computer();

    //组装产品
    @Override
    public void  BuildCPU(){
        computer.Add("1型号CPU");
    }
    @Override
    public void  BuildMainboard(){
        computer.Add("1型号主板");
    }
    @Override
    public void  BuildHD(){
        computer.Add("1型号硬盘");
    }

    //返回组装成功的电脑
    @Override
    public  Computer getComputer(){
        return computer;
    }
}
步骤3:创建具体的建造者(ConcreteBuilder):装机人员2,所用部件与人员1不同
//装机人员2
public class ConcreteBuilder2 extends Builder{
    //创建产品实例
    Computer computer = new Computer();

    //组装产品
    @Override
    public void  BuildCPU(){
        computer.Add("2型号CPU");
    }
    @Override
    public void  BuildMainboard(){
        computer.Add("2型号主板");
    }
    @Override
    public void  BuildHD(){
        computer.Add("2型号硬盘");
    }

    //返回组装成功的电脑
    @Override
    public  Computer getComputer(){
        return computer;
    }
}
步骤4:电脑城老板(Director)委派任务给装机人员进行电脑组装
public class Director {
    private Builder builder = null;

    public Director(Builder builder) {
        this.builder = builder;
    }

    //指挥装机人员组装电脑,按顺序构建出电脑
    public Computer build(){
        builder.BuildCPU();
        builder.BuildMainboard();
        builder.BuildHD();
        return builder.getComputer();
    }
}
步骤5:客户端调用,小成到电脑城找老板买电脑
public class Client {
    public static void main(String[] args){
        //逛了很久终于发现一家合适的电脑店
        //找到该店的老板,沟通需求后,老板叫装机人员1去装电脑(1型号比较适合小成)
        Builder builder = new ConcreteBuilder1();
        Director director = new Director(builder);
        //指挥装机人员1组装电脑
        Computer computer = director.build();
        //电脑组装完成,组装人员展示电脑给小成看
        computer.show();
    }
}
输出结果
组件1型号CPU装好了
组件1型号主板装好了
组件1型号硬盘装好了
电脑组装完成,请验收

3.优缺点

3.1优点

  • 易于解耦
    将产品本身与产品创建过程进行解耦,可以使用相同的创建过程来得到不同的产品。也就说细节依赖抽象。
  • 易于精确控制对象的创建
    将复杂产品的创建步骤分解在不同的方法中,使得创建过程更加清晰
  • 易于拓展
    增加新的具体建造者无需修改原有类库的代码,易于拓展,符合“开闭原则”。

每一个具体建造者都相对独立,而与其他的具体建造者无关,因此可以很方便地
替换具体建造者或增加新的具体建造者,用户使用不同的具体建造者即可得到不
同的产品对象。

3.2缺点

  • 建造者模式所创建的产品一般具有较多的共同点,其组成部分相似;如果产品之间的差异性很大,则不适合使用建造者模式,因此其使用范围受到一定的限制。
  • 如果产品的内部变化复杂,可能会导致需要定义很多具体建造者类来实现这种变化,导致系统变得很庞大。

4.应用场景

  • 需要生成的产品对象有复杂的内部结构,这些产品对象具备共性;
  • 隔离复杂对象的创建和使用,并使得相同的创建过程可以创建不同的产品。

注:本篇文章参考了简书作者Carson_Ho的文章,文章链接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,470评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,393评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,577评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,176评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,189评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,155评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,041评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,903评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,319评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,539评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,703评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,417评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,013评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,664评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,818评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,711评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,601评论 2 353

推荐阅读更多精彩内容