构造者模式(Builder Pattern)

构造者模式(Builder Pattern)主要用于复杂对象的构建,将一个复杂对象的构造过程和它的表现层分离开来。

直接撸个盖房子的例子。
这里三个类:


构建者例子用到的三个类

BuildingDesign:房屋设计图,包括一些房屋的属性,比如宽、高、颜色。
House:最后要创建出实例的类。
HouseBuilder:房屋构建者,可以理解为一个公司,或一群工人等等,总之用户通过这个类去构建House,而不是直接自己去创建House。
贴一下各类的实现:

/**
 * 房屋设计图
 */
public class BuildingDesign {

    private double width;

    private int height;

    private String color;

    public BuildingDesign() {
    }

    public double getWidth() {
        return width;
    }

    public void setWidth(double width) {
        this.width = width;
    }

    public int getHeight() {
        return height;
    }

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

    public String getColor() {
        return color;
    }

    public void setColor(String color) {
        this.color = color;
    }

}
/**
 * 大House
 */
public class House {

    private double width;

    private int height;

    private String color;

    public House(BuildingDesign design) {
        this.width = design.getWidth();
        this.height = design.getHeight();
        this.color = design.getColor();
    }

    @Override
    public String toString() {
        return "House{" +
                "width=" + width +
                ", height=" + height +
                ", color='" + color + '\'' +
                '}';
    }
}
/**
 * 构建者
 */
public class HouseBuilder {

    private BuildingDesign buildingDesign;

    public HouseBuilder() {
        buildingDesign = new BuildingDesign();
    }

    public void setWidth(double width) {
        this.buildingDesign.setWidth(width);
    }

    public void setHeight(int height) {
        this.buildingDesign.setHeight(10);
    }

    public void setColor(String color) {
        this.buildingDesign.setColor(color);
    }

    public House build() {
        if (buildingDesign == null){
            return null;
        }else {
            return new House(buildingDesign);
        }
    }
}

使用:

public class Main {

    public static void main(String[] args) {

        HouseBuilder builder = new HouseBuilder();
        builder.setWidth(130.00);
        builder.setHeight(3);
        builder.setColor("红色");

        House house = builder.build();
        System.out.println(house);
    }
}

运行结果:

House{width=130.0, height=10, color='红色'}

这样就完成了构建者模式,这里的好处:

  1. 在调用builder.build();之前,随便房屋属性并不会造成资源浪费,因为都是在new对象之前。
  2. 屏蔽了房屋创建的复杂过程,这一点代码里没体现出来,房屋的构造过程可能是很复杂的,但用户只提供了宽、高、颜色,又或者是用户根本没有要求颜色,所以设计图里,也就是BuildingDesign里要提供默认值,所以代码修改这样:
/**
 * 房屋设计图
 */
public class BuildingDesign {

    private double width = 2;

    private int height = 100;

    private String color = "白色";
    
    ...
}

这样即使使用者不指定某些属性,也会用默认的创建出来,比如一些复杂的属性,用户可能大多数不需要设置,但是用户想设置的时候必须可以设置。

为什么跟看到的一些构建者不一样?比如OkHttp里的OkHttpClient和Request的创建。
那些只是使用了链式的表达方式,这里简单数一下链式的思想,按上面的代码:

        HouseBuilder builder = new HouseBuilder();
        builder.setWidth(130.00);
        builder.setHeight(3);
        builder.setColor("红色");
        builder.build();

比如builder是个包工头,对应上面5行代码描述起来是这样的:

        1.“包工头,过来一下”
        2.“包工头,我的房子要宽130”
        3.“包工头,我的房子要高3层”
        4.“包工头,我的房子要红色”
        5.“包工头,开始盖吧”

链式的书写方式是这样的:

        new HouseBuilder().setWidth(130.00).setHeight(3).setColor("红色").build();

这意思就行,你把包工头喊过来之后,他回说:“在你交代完成之前,我一直在,你不需要喊我,直接说你的需求”,描述起来:

        “包工头,过来一下,我的房子要宽130,高3层,红色,开始盖吧”

实现这种书写方式也很简单,设置完属性把自己返回回去就好了,修改HouseBuilder的setter方法:

/**
 * 构建者
 */
public class HouseBuilder {

    ...

    public HouseBuilder setWidth(double width) {
        this.buildingDesign.setWidth(width);
        return this;
    }

    public HouseBuilder setHeight(int height) {
        this.buildingDesign.setHeight(10);
        return this;
    }

    public HouseBuilder setColor(String color) {
        this.buildingDesign.setColor(color);
        return this;
    }

    ...
}

代码完成,一般还会把Builder设置成内部类,都一个意思,随便怎么搞。

非常的丝滑:

        System.out.println( new HouseBuilder().setWidth(130.00).setHeight(3).setColor("红色").build());
        System.out.println( new HouseBuilder().setHeight(3).setColor("红色").build());
        System.out.println( new HouseBuilder().setColor("红色").build());
        System.out.println( new HouseBuilder().build());
House{width=130.0, height=10, color='红色'}
House{width=2.0, height=10, color='红色'}
House{width=2.0, height=100, color='红色'}
House{width=2.0, height=100, color='白色'}

随便设置什么属性,或者不设置,盖住来的房子都可以住。

项目地址

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

推荐阅读更多精彩内容