靜態(tài)工廠方法和構(gòu)造器的共同缺點或者局限性就是:對于存在大量可選參數(shù)的類中罩息,其擴展性都不是很好。構(gòu)建器(Builder)可以比較好的解決這些問題茫经。
重疊構(gòu)造器
在介紹構(gòu)建器之前先介紹下重疊構(gòu)造器吧巷波。什么是重疊的構(gòu)造器呢?就是有多個構(gòu)造器科平,重疊嘛褥紫。。瞪慧。這些構(gòu)造器有一些特點:至少有一個包含所有必要屬性的構(gòu)造器髓考,必要屬性。弃酌。氨菇。這么說還有非必要屬性,是的妓湘,重疊構(gòu)造器應(yīng)用的一個場景就是查蓉,對于一個類來說,它其中的所有屬性不一定都是必須要使用或者設(shè)置值的榜贴,舉個栗子來說吧豌研,用戶在某個網(wǎng)站上注冊的時候,有一些信息比如用戶名密碼是必須要填的唬党,而有一些信息比如生日住址之類的就是非必須的鹃共。
回到正題,除了包含所有屬性的構(gòu)造器之外驶拱,還有多個構(gòu)造器霜浴,這些構(gòu)造器中至少包含必要屬性和若干個非必要屬性,也可以只包含必要屬性哦蓝纲。這些構(gòu)造器其實最終都是轉(zhuǎn)到了包含所有屬性的構(gòu)造器阴孟,只是那些未包含的非必要屬性被設(shè)置成默認值了而已晌纫。光說不練非好漢,下面就直接來個栗子說明一下吧:
public class Person {
private String name;//姓名或者用戶名->必須
private String phone;//電話->必須
private int age;//年齡->可選
private String address;//地址->可選
/**
* 包含所有屬性的構(gòu)造方法
*/
public Person(String name, String phone, int age, String address) {
this.name = name;
this.phone = phone;
this.age = age;
this.address = address;
}
/**
* 包含必要屬性和一個可選屬性的構(gòu)造方法
*/
public Person(String name, String phone, int age) {
this(name, phone, age, "");
}
/**
* 包含必要屬性和一個可選屬性的構(gòu)造方法
*/
public Person(String name, String phone, String address) {
this(name, phone, 0, address);
}
/**
* 包含必須屬性的構(gòu)造方法
*/
public Person(String name, String phone) {
this(name, phone, 0);
// this(name,phone,"");//當然也可以選擇這個構(gòu)造方法永丝,結(jié)果是一樣的
}
}
重疊構(gòu)造器的問題:
與該方法的特點對應(yīng)锹漱,如果一個類中包含很多的參數(shù),就會寫很多的構(gòu)造方法类溢,而且閱讀性不高凌蔬。還有啊,客戶端在調(diào)用的時候需要仔細了解各個構(gòu)造方法的含義闯冷,如果不小心把參數(shù)的含義看錯了砂心,或者位置顛倒了就會產(chǎn)生錯誤,而且錯誤還不好定位蛇耀。辩诞。。下面介紹另外一種方法--JavaBean模式纺涤。
JavaBean模式
這種模式也很常用译暂,簡單來說就是類中存在一個無參的構(gòu)造方法(哎呀,參考資料中都是構(gòu)造器撩炊,本人比較習慣用構(gòu)造方法外永,所以可能一會構(gòu)造器一會構(gòu)造方法的,請諒解拧咳。伯顶。。)骆膝,然后構(gòu)造方法就沒有別的了祭衩,對,就是沒有別的了阅签。你可能會問掐暮,那類中的參數(shù)怎么辦呢?別急政钟,類中還存在一大堆set方法路克,作用就是設(shè)置參數(shù)。所以养交,這種方法比較好地彌補了重疊構(gòu)造器的不足衷戈,創(chuàng)建實例的過程:
Person person=new Person();
person.setName("路飛");
person.setAge(18);
//其他的set方法。层坠。。刁笙。
同樣的破花,這種方法也存在缺點:并發(fā)問題谦趣,構(gòu)造過程中JavaBean可能處于不一致的狀態(tài)。并且座每,該方法阻止了把類做成不可變的可能前鹅。
前面巴拉巴拉一大堆,目的就是為了引出本文的重頭戲:Builder模式峭梳。
Builder模式
上代碼:
public class PersonB {
private String name;//姓名或者用戶名->必須
private String phone;//電話->必須
private int age;//年齡->可選
private String address;//地址->可選
/**
* 包含所有屬性的構(gòu)造方法
*/
private PersonB(Builder builder) {//通過Builder來構(gòu)造舰绘,該方法是private類型的
this.name = builder.name;
this.phone = builder.phone;
this.age = builder.age;
this.address = builder.address;
}
public static class Builder {
private String name;//姓名或者用戶名->必須
private String phone;//電話->必須
//設(shè)置了默認值
private int age = 0;//年齡->可選
private String address = "";//地址->可選
public Builder(String name, String phone) {
this.name = name;
this.phone = phone;
}
public Builder age(int age) {
this.age = age;
return this;
}
public Builder address(String address) {
this.address = address;
return this;
}
public PersonB build() {
return new PersonB(this);
}
}
}
注意:PersonB這個類的構(gòu)造方法是private類型的,也就是說無法通過new PersonB()來創(chuàng)建實例葱椭。具體怎么用呢捂寿,來來來,敲黑板啦~
PersonB person = new PersonB.Builder("山治", "110").age(20).address("桑尼號").build();
是不是很簡單孵运,而且也很明了秦陋,傳遞了必要的參數(shù)和非必須的參數(shù)。同時在build()方法中可以對參數(shù)進行檢驗(你需要根據(jù)實際情況添加對應(yīng)的檢驗邏輯)治笨。
在抽象工廠中的應(yīng)用:
客戶端可以將builder傳遞給某個方法驳概,這樣該方法就能夠創(chuàng)建一個或者多個對象了。如果可以使用泛型的話:
public interface Builder<T> {
public T build();
}
Tree buildeTree(Builder<? extends Node> nodeBuilder) { ... }
代碼中第二部分就是一個實際的應(yīng)用旷赖,功能就是通過Builder來創(chuàng)建一棵樹顺又。。等孵。
下面說下該方法的缺點吧:
一般來說一個方法的缺點往往與其實現(xiàn)方法相關(guān)稚照,Builder模式的缺點就是,必須要先創(chuàng)建它的構(gòu)建器Builder(這不是廢話嗎)流济。該方法可能更加冗長锐锣,比較適合參數(shù)多的情況,比如有4個或者更多的參數(shù)绳瘟。
總之:如果類的構(gòu)造器或者靜態(tài)工廠中有多個參數(shù)雕憔,在設(shè)計這種類的時候,可以考慮Builder模式糖声,當一個類中大多數(shù)參數(shù)都是可選的時候就更完美了~斤彼。