2019-04-04设计模式-Builder模式

Builder 模式:又叫建造者模式,将一个复杂对象的构建和表示分离,通过不同的构建,出现不同的表示;解耦代码,增加扩展性

这里主要通过分析 android 源码中的 AlertDialog 进而理解这个模式;先来看简单使用

    new AlertDialog.Builder(this)
                .setTitle("测试")
                .setMessage("测试对话框")
                .show();

通过 new 一个 Builder 对象,给这个对象设置不同参数,从而得到不同样式的 Dialog ;看 AlertDialog.Builder(this) 做了什么

    //这里只截取部分代码
    public static class Builder {
        //定义一个 AlertParams p 用来保存参数
        private final AlertParams P;
        private final int mTheme;

        public Builder(@NonNull Context context) {
            this(context, AlertDialog.resolveDialogTheme(context, 0));
        }

        public Builder(@NonNull Context context, @StyleRes int themeResId) {
            //在构造方法中初始化 AlertParams
            this.P = new AlertParams(new ContextThemeWrapper(context, AlertDialog.resolveDialogTheme(context, themeResId)));
            this.mTheme = themeResId;
        }
    }

主要就是 new AlertParams 对象,这个对象用来保存 set 进来的参数;如下:

        public AlertDialog.Builder setTitle(@Nullable CharSequence title) {
            this.P.mTitle = title;
            return this;
        }

setTitle 就是将 title 设置到 AlertParams 的 mTitle 中,其他 set 方法同理,都是将参数设置到 AlertParams 中,继续往下看 .show(); 方法

        public AlertDialog show() {
            //调用自身的 create 方法,创建 AlertDialog 对象
            AlertDialog dialog = this.create();
            //显示对话框
            dialog.show();
            return dialog;
        }
    
        public AlertDialog create() {
            //new 一个 AlertDialog 对象
            AlertDialog dialog = new AlertDialog(this.P.mContext, this.mTheme);
            //......
            return dialog;
        }

先来看这一部分,show 方法其实就是 new AlertDialog ,得到 AlertDialog 对象,先来看 new AlertDialog 做了什么

    
public class AlertDialog extends AppCompatDialog implements DialogInterface {
    final AlertController mAlert;
    static final int LAYOUT_HINT_NONE = 0;
    static final int LAYOUT_HINT_SIDE = 1;

    //注意看,这几个构造方法最终都会调用第二个构造方法
    protected AlertDialog(@NonNull Context context) {
        this(context, 0);
    }

    protected AlertDialog(@NonNull Context context, @StyleRes int themeResId) {
        super(context, resolveDialogTheme(context, themeResId));
        //创建一个 AlertController 对象,并将 AlertDialog 对象传过去
        this.mAlert = new AlertController(this.getContext(), this, this.getWindow());
    }

    protected AlertDialog(@NonNull Context context, boolean cancelable, @Nullable OnCancelListener cancelListener) {
        this(context, 0);
        this.setCancelable(cancelable);
        this.setOnCancelListener(cancelListener);
    }
}

所以,目前为止一共出现了四个类

  1. AlertDialog
  2. Builder
  3. AlertController
  4. AlertParams

其中 Builder 是 AlertDialog 的内部类,AlertParams 是 AlertController 的内部类;Builder 持有一个 AlertParams 对象,AlertController 持有一个 AlertDialog 对象;继续看 create 方法

        public AlertDialog create() {
            //创建 AlertDialog 对象和 AlertController 对象
            AlertDialog dialog = new AlertDialog(this.P.mContext, this.mTheme);
            //调用 AlertParams.apply 方法 并将 AlertController 传过去
            this.P.apply(dialog.mAlert);
            dialog.setCancelable(this.P.mCancelable);
            if (this.P.mCancelable) {
                dialog.setCanceledOnTouchOutside(true);
            }

            dialog.setOnCancelListener(this.P.mOnCancelListener);
            dialog.setOnDismissListener(this.P.mOnDismissListener);
            if (this.P.mOnKeyListener != null) {
                dialog.setOnKeyListener(this.P.mOnKeyListener);
            }

            return dialog;
        }

关键就是 AlertParams.apply 方法:

    //这是删减后的代码
    public void apply(AlertController dialog) {
        //判断是否设置有 mCustomTitleView
        if (this.mCustomTitleView != null) {
                dialog.setCustomTitle(this.mCustomTitleView);
            } else {
                //判断 mTitle 是否为null
                if (this.mTitle != null) {
                    //我们前面设置了,所以不为null
                    //将 mTitle 设置给 dialog
                    //注意 dialog 对象是一个 AlertController
                    dialog.setTitle(this.mTitle);
                }
          }
    }

apply 方法主要就是判断我们 set 了那些参数,进行非空判断之后又这是给 AlertController 对象;到目前为止,AlertController 和 AlertParams 中包含了我们 set 的参数,然后就没有了,那么是怎么设置到 AlertDialog 中,让 AlertDialog 根据不同参数显示不同效果呢?下面来看 dialog.show(); 方法

        //前面分析的 show ,现在要看的是 dialog.show();
        public AlertDialog show() {
            AlertDialog dialog = this.create();
            //来看这个方法
            dialog.show();
            return dialog;
        }

dialog.show(); 就是将 Dialog 显示到界面上

    public void show() {
        //.....
        //判断是否创建,第一次肯定是false
        if (!mCreated) {
            dispatchOnCreate(null);
        } else {
            // Fill the DecorView in on any configuration changes that
            // may have occured while it was removed from the WindowManager.
            final Configuration config = mContext.getResources().getConfiguration();
            mWindow.getDecorView().dispatchConfigurationChanged(config);
        }
      //.......
    }

    void dispatchOnCreate(Bundle savedInstanceState) {
        if (!mCreated) {
            //调用 onCreate
            onCreate(savedInstanceState);
            mCreated = true;
        }
    }
    
    //发现是一个空方法,我们回到 AlertDialog 中,看怎么重写
    protected void onCreate(Bundle savedInstanceState) {
    }

走了一圈,发现是一个空方法,我们回到 AlertDialog 中,看怎么重写

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //调用 mAlert.installContent
        // 就是 AlertController.installContent
        this.mAlert.installContent();
    }

最后发现调用了 AlertController.installContent 好像这些模块都关联起来了

    public void installContent() {
        int contentView = this.selectContentView();
        this.mDialog.setContentView(contentView);
        this.setupView();
    }
    
    private void setupView() {
        //.........
        this.setupTitle(topPanel);
        //......
    }

    private void setupTitle(ViewGroup topPanel) {
        View titleTemplate;
        //判断 mCustomTitleView 是否为null
        if (this.mCustomTitleView != null) {
            LayoutParams lp = new LayoutParams(-1, -2);
            topPanel.addView(this.mCustomTitleView, 0, lp);
            titleTemplate = this.mWindow.findViewById(id.title_template);
            titleTemplate.setVisibility(8);
        } else {
            this.mIconView = (ImageView)this.mWindow.findViewById(16908294);
            boolean hasTextTitle = !TextUtils.isEmpty(this.mTitle);
            if (hasTextTitle && this.mShowTitle) {
                //找到默认 titleView
                this.mTitleView = (TextView)this.mWindow.findViewById(id.alertTitle);
                //设置标题
                this.mTitleView.setText(this.mTitle);
                if (this.mIconId != 0) {
                    this.mIconView.setImageResource(this.mIconId);
                } else if (this.mIcon != null) {
                    this.mIconView.setImageDrawable(this.mIcon);
                } else {
                    this.mTitleView.setPadding(this.mIconView.getPaddingLeft(), this.mIconView.getPaddingTop(), this.mIconView.getPaddingRight(), this.mIconView.getPaddingBottom());
                    this.mIconView.setVisibility(8);
                }
            } else {
                titleTemplate = this.mWindow.findViewById(id.title_template);
                titleTemplate.setVisibility(8);
                this.mIconView.setVisibility(8);
                topPanel.setVisibility(8);
            }
        }

    }

最后发现,根据不同的set然后给Dialog设置了不同的样子

小总结

当调用 Builder.create() 方法时,就是将各种参数设置进去,就和建房子准备砖头水泥;当调用dialog.show() 就是构建需要的对象,就和建房子开始盖。最后在附上UML图


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

推荐阅读更多精彩内容