美文网首页设计模式
设计模式-Builder模式

设计模式-Builder模式

作者: i卓 | 来源:发表于2017-09-25 11:33 被阅读16次

    上一篇文章设计模式-单例模式我们介绍了单例模式的几种用法和优缺点,具体的我们需要结合项目中的场景去具体选择。这一篇我们来学习Builder模式(有的翻译为建造者、构建者),builder模式在Android的开发场景中用的还是挺多的,比如Android源码中的AlertDialog、StringBuilder和StringBuffe等等,三方库中的GsonBuilder、EventBus中的builder等等,太常见了,有的人把链式调用和builder设计模式搞混,其实这两者是完全不同的,Builder设计模式是设计模式的一种,链式调用是一种函数调用方式。

    Builder设计模式:

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

    上面这句话摘自网络,说的比较官方,不懂的人还是一头雾水。没关系下面我们来举例说明Builder设计模式是怎么创建出来的。
    咱们还是拿一个经典的Person类来讲解,我们定义一个Person类有姓名、年龄、身高、体重四个基本的属性:

    public class Person {
        private String name;
        private int age;
        private double height;
        private double weight;
    
        //为了篇幅省略get、set方法
    }
    

    我们可能为了能够方便的 new 一个对象,定义几个构造方法:

    public Person() {
    }
    
    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, double height) {
         this.name = name;
         this.age = age;
         this.height = height;
    }
    
    public Person(String name, int age, double height, double weight) {
            this.name = name;
            this.age = age;
            this.height = height;
            this.weight = weight;
    }
    

    可以看到,我们定义了几个构造方法,包括无参、一个参数、两个参数的等等。
    然后我们就可以这样创建多个Person:

    Person p1=new Person();
    Person p2=new Person("张三");
    Person p3=new Person("李四",18);
    Person p4=new Person("王五",21,180);
    Person p5=new Person("赵六",17,170,65.4);
    

    但是我们发现一个问题,参数一多,我们根本不清楚参数代表的具体意义,传参的时候可能需要点到源码中查看相关释义。还有就是当这个类中需要传入的参数很多的时候,构造方法就会特别多,这样对我们的开发和维护是很不利的,有没有方法可以解决这个问题呢?当然,看看我们运用Builder模式是怎么处理这段代码的:

    public class Person {
        private String name;
        private int age;
        private double height;
        private double weight;
    
        private Person(Builder builder) {
            this.name = builder.name;
            this.age = builder.age;
            this.height = builder.height;
            this.weight = builder.weight;
        }
    
        //一大堆get、set
    
        static class Builder{
            private String name;
            private int age;
            private double height;
            private double weight;
    
            public Builder setName(String name){
                this.name = name;
                return this;
            }
    
            public Builder setAge(int age){
                this.age = age;
                return this;
            }
    
            public Builder setHeight(double height){
                this.height = height;
                return this;
            }
    
            public Builder setWeight(double weight){
                this.weight = weight;
                return this;
            }
    
            public Person build(){
                return new Person(this);
            }
        }
    }
    

    从上面的代码我们可以看到:

    1.我们创建了一个静态的Builder内部类;

    2.我们定义了和Person类中一样的成员变量;

    3.Builder中的成员函数来给这些变量赋值,同时返回builder本身;

    4.同时提供一个build函数来创建一个Person对象;

    5.在Person对象中存在一个私有的构造函数来传入该builder对象依次给Person对象自己的成员变量赋值。

    这里有几点需要说明:

    Builder是static修饰的:

    静态的内部类不持有外部内的引用,如果Builder依赖于外部类,那么Builder模式就毫无意义可言;

    Person的构造函数用private修饰:

    Person的创建由Builder提供,无需对外暴露;

    关于链式调用:

    Builder成员函数返回Builder对象本身,这是为了支持链式调用,而链式调用是非必需的,使用链式调用一般是为了增加易用性和代码的可读性。

    Builder在Android中的体现

    常用的Gson解析创建一个Gson对象,比如在我的项目中

    Gson gson = new GsonBuilder().registerTypeAdapterFactory(new NullStringToEmptyAdapterFactory()).create();
    

    多说一句:这里的NullStringToEmptyAdapterFactory是为了解决服务端返回null,客户端直接处理导致崩溃的情况

    其它的还有StringBuilder、Okhttp、EventBus、Glide等等,有兴趣的话可以去看一看她们的源码。好了,Builder模式我们讲完了,接下来的设计模式系列我将来讲讲工厂模式,敬请期待。

    相关文章

      网友评论

        本文标题:设计模式-Builder模式

        本文链接:https://www.haomeiwen.com/subject/hknnsxtx.html