设计模式之Builder模式

  Builder设计模式在代码设计中很常见,譬如我们在阅读大神们的源码是,一些初始化参数比较多构造方法都采用了Builder模式,譬如说Okhttp,Glide,Picasso

Builder模式长什么样?

  举个简单的例子,现在大多数Android应用的开发网络层大多数都在使用OkHttp,熟悉的人都知道,在使用Okhttp之前要进行一些初始化工作,譬如说超时时间缓存策略等等,下面是个简单的初始化例子:

 public static OkHttpClient get() {
        if (okHttpClient == null) {
            synchronized (AppHttpClient.class) {
                if (okHttpClient == null) {
                    okHttpClient = new OkHttpClient.Builder()
                            //cookie保存
                            .cookieJar(new CookiesManager(new PersistentCookieStore(BaseApplication.getInstance())))
                            .sslSocketFactory(createSSLSocketFactory())
                            //支持SSL
                            .hostnameVerifier(new HostnameVerifier() {
                                                  @Override
                                                  public boolean verify(String hostname, SSLSession session) {
                                                      return true;
                                                  }
                                              }
                            )
                            //链接15S超时
                            .connectTimeout(15, TimeUnit.SECONDS)
                            //Read请求30S超时
                            .readTimeout(30, TimeUnit.SECONDS)
                            //Write请求30S超时
                            .writeTimeout(30, TimeUnit.SECONDS)
                            //增加Header
                            .addInterceptor(new HeaderInfoInterceptor())
                            //build
                            .build();
                }
            }
        }
        return okHttpClient;
    }

上面的链式调用方法进行初始化使用的就是Builder设计模式,是不是觉得很方便?

使用场景

  Builder可以将复杂对象的建造过程抽象出来(抽象类别),使这个抽象过程的不同实现方法可以构造出不同表现(属性)的对象。这句话看起来很拗口,不好理解,下边我们举一个简单的例子:

我们在实际开发过程当中,时常会遇到这样一个情况,需要构建一个复杂(属性N多)的对象,像这样婶儿的:

public class Person {
    private String name;        //必须
    private int age;            //必须
    private int sex;            //必须
    private float height;
    private String phone;
    private String address;
    private String email;
}

想要new一个这样的类的实例,于是我们就开始撸下边的代码,通过构造函数的参数的方式去new一个对象:

public Person(String name) {
        this.name = name;
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Person(String name, int age, float height) {
        this.name = name;
        this.age = age;
        this.height = height;
    }

    public Person(String name, int age, float height, int sex) {
        this.name = name;
        this.age = age;
        this.height = height;
        this.sex = sex;
    }

    public Person(String name, int age, float height, int sex, String phone) {
        this.name = name;
        this.age = age;
        this.height = height;
        this.sex = sex;
        this.phone = phone;
    }

    public Person(String name, int age, float height, int sex, String phone, String address) {
        this.name = name;
        this.age = age;
        this.height = height;
        this.sex = sex;
        this.phone = phone;
        this.address = address;
    }

    public Person(String name, int age, float height, int sex, String phone, String address, String email) {
        this.name = name;
        this.age = age;
        this.height = height;
        this.sex = sex;
        this.phone = phone;
        this.address = address;
        this.email = email;
    }

或者使用getter和setter的方式去写一个实现类:

public class Person {
    private String name;        //必须
    private int age;            //必须
    private int sex;            //必须
    private float height;
    private String phone;
    private String address;
    private String email;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public int getSex() {
        return sex;
    }

    public void setSex(int sex) {
        this.sex = sex;
    }

    public float getHeight() {
        return height;
    }

    public void setHeight(float height) {
        this.height = height;
    }

    public String getPhone() {
        return phone;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }
}

先说说这两种方式的优劣:

第一种在参数不多的情况下,是比较方便快捷的,一旦参数多了,代码可读性大大降低,并且难以维护,对调用者来说也造成一定困惑;

第二种可读性不错,也易于维护,但是这样子做对象会产生不一致的状态,当你想要传入全部参数的时候,你必需将所有的setXX方法调用完成之后才行。然而一部分的调用者看到了这个对象后,以为这个对象已经创建完毕,就直接使用了,其实Person对象并没有创建完成,另外,这个Person对象也是可变的,不可变类所有好处都不复存在。

所以有没有更好地方式去实现它呢,那就是接下来要理解的Builder模式了。

以下介绍的Builder模式其实是Builder模式的衍生模式,他与最经典Builder模式有区别,因为现在很少使用经典Builder模式,基本都在使用Builder衍生模式(链式调用),所以经典Builder模式本人也没怎么研究,有兴趣的童鞋可以自行研究下设计模式之四 --- 建造(Builder)模式

新撸出来的代码:

public class Person {
    private final String name;        //必须
    private final int age;            //必须
    private final int sex;            //必须
    private final float height;
    private final String phone;
    private final String address;
    private final String email;


    private Person(PersonBuilder builder){
        this.name = builder.name;
        this.age = builder.age;
        this.sex = builder.sex;
        this.height = builder.height;
        this.phone = builder.phone;
        this.address = builder.address;
        this.email = builder.email;
    }

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

    public int getSex() {
        return sex;
    }

    public float getHeight() {
        return height;
    }

    public String getPhone() {
        return phone;
    }

    public String getAddress() {
        return address;
    }

    public String getEmail() {
        return email;
    }


    public static class PersonBuilder{
        private final String name;        //必须
        private final int age;            //必须
        private final int sex;            //必须
        private  float height;
        private  String phone;
        private  String address;
        private  String email;

        public PersonBuilder(String name, int age, int sex) {
            this.name = name;
            this.age = age;
            this.sex = sex;
        }

        public PersonBuilder height(float height){
            this.height = height;
            return this;
        }

        public PersonBuilder phone(String phone){
            this.phone = phone;
            return this;
        }

        public PersonBuilder address(String address){
            this.address = address;
            return this;
        }

        public PersonBuilder email(String email){
            this.email = email;
            return this;
        }

        public Person build(){
            return new Person(this);
        }
    }
}

从上面的代码中我们可以看到,我们在PersonBuilder类里定义了一份与Person一模一样的变量,通过一系列的成员函数进行设置属性值,但是返回值都是this,也就是都是PersonBuilder对象,最后提供了一个build函数用于创建Person对象,返回的是Person对象,对应的构造函数在Person类中进行定义,也就是构造函数的入参PersonBuilder对象,然后依次对自己的成员变量进行赋值,对应的值都是PersonBuilder对象中的值。此外PersonBuilder类中的成员函数返回Builder对象自身的另一个作用就是让它支持链式调用,使代码可读性大大增强。

Builder模式需要有几个注意的点:

  1. Person类的构造方法必须是私有的,调用者不能直接创建Person对象
  2. Person类的属性是不可变的,所有的属性都要添加final修饰符,并且在构造方法中设置了值,并且对外只提供getters方法。
  3. PersonBuilder的内部类构造方法中只接收必传的参数,并且必传的参数使用final修饰符。

调用方式

new Person.PersonBuilder("张三", 18, 1)
                .address("北京市XXX")
                .email("xxxxx@gmail.com")
                .height(175.5f)
                .phone("1234556")
                .build();

相比起前面通过构造函数和setter/getter方法两种方式,可读性更强。唯一可能存在的问题就是会产生多余的Builder对象,消耗内存。然而大多数情况下我们的Builder内部类使用的是静态修饰的(static),所以这个问题也没多大关系。

怎么样Builder设计模式会使你的代码看上去高大上一些呢?

Android Studio对Builder模式的支持

AS 对Builder这种常用的设计模式提供了支持,方便我们的代码编写,在Plugins中搜索Builder,会出现一个InnerBuilder的插件,

1.jpeg

在编写好类和属性之后,右键generate选择Builder,选择属性

2.jpeg

确定之后就会生成Builder模式下的代码,当然生成的代码和我们希望的还是有些不同,只需要进行小小的修改就可以了,还是很方便的!

总结

优点

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

缺点

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

推荐阅读更多精彩内容