JAVA设计模式【创建型模式】之【Builder】

前言

其他创建型模式:

今天来介绍创建型模式之Builder(生成器)。Builder模式在Android开发中我们遇到很多,无论是平时的开发,还是在阅读Android源码的过程都会遇到。例如Android源码中:

  • app核心层: Notification.Builder(android.app)
  • 图形绘制层:TypeFace.Builder(android.graphics)
  • 底层硬件层:BluetoothDeviceFilter.Builder(android.companion)
  • 网络层:Uri.Builder(android.net)

可以发现,在Android的源码框架中,Builder模式的运用真的是非常普遍。

大家可以通过快捷键在Android源码当中搜索Builder即可。最常见的就是AlertDialog.Builder对象,如下图所示:

Builder

可以看到这么多方法才使得我们在coding时去create一个Dialog对象的时候才能随心所欲,Android中创建Dialog的这种方式,充分的体现了Builder模式的强大和灵活。

接下来我们来具体学习Builder模式:

1. 意图

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

2. 适用性

  • 当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时。
  • 当构造过程必须允许被构造的对象有不同的表示时。

3. 参与者

  • Builder —— 为创建一个Product对象的各个部件指定抽象接口。
  • ConcreteBuilder
    —实现Builder的接口以构造和装配该产品的各个部件。
    — 定义并明确它所创建的表示。
    — 提供一个检索产品的接口。
  • Director —— 构造一个使用Builder接口的对象。
  • Product
    — 表示被构造的复杂对象。ConcreteBuilder创建该产品的内部表示并定义它的装配过程。
    — 包含定义组成部件的类,包含将这些部件装配成最终产品的接口。

4. 效果

1)它使你可以改变一个产品的内部表示时
2)它将构造代码和表示代码分开
3)它使你对构造过程进行更精细的控制

5. 实例

我们还是以汽车为例子,


汽车构造图

汽车一般一共由五大部分组成:

  • 发动机(Engine)
  • 传动系统(Power)
  • 车身(body)
  • 转向(Steering)
  • 悬挂(Suspension)

那么就可以抽象出对应的Builder接口:
CarBuilder.java

  public interface CarBuilder {
    /**
     * 构建发动机
     */
    void buildEngine();
    /**
     * 构建传动系统
     */
    void buildPower();
    /**
     * 构建车身
     */
    void buildBody();
    /**
     * 构建转向
     */
    void buildSteering();
    /**
     * 构建悬挂
     */
    void buildSuspension();

    Car buildCar();

那么对应的参与者Product就是汽车本身,这里我们以宝马和奥迪来进行举例:

Car汽车类图

对应的ConcreteBuilder 就是AudiBuilder和BmwBuilder,如下图
image.png

然后需要一个Director的角色来指导不同的Car的构建:

CarDirector.java

public class CarDirector {
    public Car buildCar(CarBuilder carBuilder) {
        carBuilder.buildBody();
        carBuilder.buildEngine();
        carBuilder.buildPower();
        carBuilder.buildSteering();
        carBuilder.buildSuspension();
        return carBuilder.buildCar();
    }
}

最后我们来测试一下:

private static void testBuilder() {
        CarDirector carDirector = new CarDirector();
        //构建奥迪车
        final Car audiCar = carDirector.buildCar(new AudiCarBuilder());
        System.out.println(audiCar.toString());
        //构建宝马车
        final Car bmwCar = carDirector.buildCar(new BmwCarBuilder());
        System.out.println(bmwCar.toString());
 }

测试结果下图所示:


image.png

实践完汽车的这个简单的实例,我们再来看一下Android中经典的Builder模式之AlertDialog.Builder:

//很经典的链式调用有木有(一气呵成)
AlertDialog dialog = new AlertDialog.Builder(this)
                .setTitle("title")
                .setMessage("message")
                .create();
            dialog.show();

通过阅读AlertDialog.Builder源码,可以看出Builder类是属于AlertDialog类的一个静态内部类,由源码过多,我们只分析部分构建过程:

public class AlertDialog extends Dialog implements DialogInterface {

    //.....此处省略若干行代码

    public static class Builder {
        private final AlertController.AlertParams P;

        public Builder(Context context) {
            this(context, resolveDialogTheme(context, ResourceId.ID_NULL));
        }

        public Builder setTitle(@StringRes int titleId) {
            P.mTitle = P.mContext.getText(titleId);
            return this;
        }


        public Builder setMessage(CharSequence message) {
            P.mMessage = message;
            return this;
        }

        public AlertDialog create() {
            //构建一个新AlertDialog对象,并通过AlertController.AlertParams配置Dialog
            final AlertDialog dialog = new AlertDialog(P.mContext, 0, false);
            P.apply(dialog.mAlert);
            dialog.setCancelable(P.mCancelable);
            if (P.mCancelable) {
                dialog.setCanceledOnTouchOutside(true);
            }
            dialog.setOnCancelListener(P.mOnCancelListener);
            dialog.setOnDismissListener(P.mOnDismissListener);
            if (P.mOnKeyListener != null) {
                dialog.setOnKeyListener(P.mOnKeyListener);
            }
            return dialog;
        }

        public AlertDialog show() {
            final AlertDialog dialog = create();
            dialog.show();
            return dialog;
        }
        
        //.....此处省略若干行代码
    }
}

和构造汽车的Builder模式相比较,AlertDialog.Builder缺少了Director角色,但是这并不影响Builder模式的效果,其精髓之处就是体现在程序员一步步构建Dialog对象的时候,其实也可以把具体使用Dialog情景或者方法体看成是一个虚的Director

第三方库中Builder的使用

另外Builder的构建经常在静态工厂的构建对象或者作为第三方库使用的时候出现,比如知乎开源的一个强大图片选择器Matisse


在使用Matisse的时候就是经典的链式调用+Builder模式,这里Matisse本身其实就是Builder !!!

Matisse.from(MainActivity.this)
        .choose(MimeType.allOf())
        .countable(true)
        .maxSelectable(9)
        .addFilter(new GifSizeFilter(320, 320, 5 * Filter.K * Filter.K))
        .gridExpectedSize(getResources().getDimensionPixelSize(R.dimen.grid_expected_size))
        .restrictOrientation(ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED)
        .thumbnailScale(0.85f)
        .imageEngine(new GlideEngine())
        .forResult(REQUEST_CODE_CHOOSE);

大家有兴趣可以去阅读以下它的源码,另外这里面还应用了其他模式,例如代理模式、模板方法等,这里会在后续的文章进行介绍。

6. 总结

Abstract FactoryBuilder相似,因为它也可以创建复杂对象。主要的区别就是Builder模式着重于一步步构造一个复杂的对象。而Abstract Factory着重于多个系列产品的对象的构建。Builder在最后一步产品返回的时候,对于Abstract Factory来说,产品是立即返回的。

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

推荐阅读更多精彩内容