设计模式之Builder模式

Builder模式介绍

Builder模式是一步一步创建一个复杂对象的创建型模式,运行用户在不知道的内部构建细节的情况下,可以精细地控制对象的构造流程。该模式是为了构建复制对象的过程和它的部件解耦,使得构建过程和部件的表示隔离开来。

Builder模式定义

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

Builder模式的使用场景

  • 相同的方法,不同的执行顺序,产生不同的事件结果。
  • 多个部件,都可以装配到一个对象中,但产生的运行结果不同。
  • 产品类非常复杂,或者产品类中的调用顺序不同产生了不同的作用,这个时候使用建造者模式非常合适。
  • 初始化一个对象特别复杂,如参数多,且很多参数都具有默认值。

Builder模式的UML类图

角色介绍:

  • Product产品类---产品的抽象类;
  • Builder---抽象Builder 类,规范产品的组建,一般由子类实现具体的组建过程;
  • ConcreteBuilder---具体的Builder类;
  • Direcor ---统一组装过程。
Builder.png

Builder模式的简单实现

计算机的组装过程较为复杂,并且组装顺序不固定,为了易于理解,我们把计算机组装的过程简化为构建主机、设置操作系统、设置显示器3 个部分,然后通过Director和具体的Builder 来构建计算机对象。代码如下:

/**
 * 计算机抽象类,即Produce角色
 */
public abstract class Computer {
    protected String mBoard;
    protected String mDisplay;
    protected String mOs;

    public void setmBoard(String mBoard) {
        this.mBoard = mBoard;
    }

    public void setmDisplay(String mDisplay) {
        this.mDisplay = mDisplay;
    }

    public abstract void setmOs();

    @Override
    public String toString() {
        return "Computer{" +
                "mBoard='" + mBoard + '\'' +
                ", mDisplay='" + mDisplay + '\'' +
                ", mOs='" + mOs + '\'' +
                '}';
    }
}

/**
 * 具体的Computer类
 */
public class MacBook extends Computer {
    @Override
    public void setmOs() {
        mOs = "Mac OS";
    }
}

/**
 * 抽象Builder类
 */
public abstract class Builder {
    public abstract void builderBoard(String board);
    public abstract void builderDisplay(String display);
    public abstract void builderOS();
    public abstract Computer create();
}


/**
 * 具体的Builder类,MacBookBilder
 */
public class MacbookBuilder extends Builder {
    private Computer mComputer = new MacBook();
    @Override
    public void builderBoard(String board) {
        mComputer.setmBoard(board);
    }

    @Override
    public void builderDisplay(String display) {
        mComputer.setmDisplay(display);
    }

    @Override
    public void builderOS() {
        mComputer.setmOs();
    }

    @Override
    public Computer create() {
        return mComputer;
    }
}


/**
 * Director类,负责构造Computer
 */
public class Director {
    Builder builder = null;

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

    public void construct(String broad,String dispplay){
        builder.builderBoard(broad);
        builder.builderDisplay(dispplay);
        builder.builderOS();
    }
}

/**
 * 测试
 */
 @Test
 public void test(){
     //构建器
     MacbookBuilder builder = new MacbookBuilder();

     Director director = new Director(builder);
     //封装构建过程
     director.construct("英特尔主板","Retina显示器");
     System.out.println(builder.create().toString());
  }

/**
 *输出结果 Computer{mBoard='英特尔主板', mDisplay='Retina显示器', mOs='Mac OS'}
 */

上述示例中,通过具体的MacbookBuilder 来构建MacBook对象,而Director封装了构建复杂产品对象的过程,对外隐藏构建细节。BuilderDirector 一起将一个复杂的构建与它的表示分离,使得同样的构建过程可以创建不同的对象。

在实际开发中,Director 角色经常会被省略,而直接使用一个Builder来进行对象的组装,这个Builder 通常为链式调用,它的关键点是每个setter方法都返回自身,也就是return this 这样就使得setter方法可以链式调用,代码如下:

        Computer computer = new MacbookBuilder()
                .setBoard("英特尔")
                .setDisplay("Retina显示器")
                .setOS()
                .create();
        System.out.println(computer.toString());

通过这种方式不仅去除了Director 角色,整个结构也更加简单,也能对Product 对象的组装过程有更加精细的控制。


总结

Builder 模式在Android 开发中通常作为配置累的构建器将配置的构建和表示分离开来,同时也将配置从目标中隔离出来,避免过多的setter 方法,Builder 模式比较常见的实现形式是通过链式实现,这样代码更简洁,易懂。

优点

  • 良好的封装性,使用建造者模式可以是客户端不必知道产品内部组成的细节。
  • 建造者独立,容易扩展。

缺点

  • 会产生多余的Builder 对象以及Direator 对象,消耗内存。

Demo

设计模式Demo

参考

《Android源码设计模式》

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

推荐阅读更多精彩内容