Builder模式是安卓開發(fā)中一種常見的設(shè)計(jì)模式,這里我們簡單介紹一下磁玉。
Builder的定義
將一個(gè)復(fù)雜對(duì)象的構(gòu)造與他的表示分離年碘,使得同樣的構(gòu)造過程可以創(chuàng)建不同的表示娄涩。
Builder的使用
Builder設(shè)計(jì)模式的代碼長什么樣子,先來大致看一下:
? new AlertDialog.Builder(this)
? ? ? ? ? ? ? ? .setTitle("the title")
? ? ? ? ? ? ? ? .setMessage("the message")
? ? ? ? ? ? ? ? .setCancelable(true)
? ? ? ? ? ? ? ? .create()
? ? ? ? ? ? ? ? .show();
很優(yōu)雅有木有淌哟,這個(gè)鏈?zhǔn)秸{(diào)用就是Builder一個(gè)很明顯的特征,可讀性非常強(qiáng)辽故。
我們在什么情況下適合用Builder模式呢徒仓?
當(dāng)我們要?jiǎng)?chuàng)建一個(gè)類時(shí),并且這個(gè)類中有很多屬性且有一部分是可選屬性誊垢,那么這時(shí)候我們就可以使用Builder模式掉弛。
這里先貼一個(gè)使用Builder的一個(gè)demo類症见,如下:
public class Student {
? ? private final String name;
? ? private final String age;
? ? private final String phone;
? ? private final String address;
? ? private Student(Builder builder){
? ? ? ? ????this.name=builder.name;
? ? ? ? ????this.address=builder.address;
? ? ? ????? this.age=builder.age;
? ? ? ? ????this.phone=builder.phone;
? ? }
? ? public String getName() {
? ? ? ????? return name;
? ? }
? ? public String getAge() {
? ? ? ? ????return age;
? ? }
? ? public String getPhone() {
? ? ? ????? return phone;
? ? }
? ? public String getAddress() {
? ? ? ? ????return address;
? ? }
? ? public static class Builder{
? ? ? ? private final String name;
? ? ? ? private final String age;
? ? ? ? private? String phone;
? ? ? ? private? String address;
? ? ? ? public Builder(String name, String age) {
? ? ? ? ? ? ????this.name = name;
? ? ? ? ? ? ????this.age = age;
? ? ? ? }
? ? ? ? public Builder phone(String phone){
? ? ? ? ? ? ????this.phone=phone;
? ? ? ? ? ? ????return this;
? ? ? ? }
? ? ? ? public Builder address(String address){
? ? ? ? ? ????? this.address=address;
? ? ? ? ? ? ????return this;
? ? ? ? }
? ? ? ? public Student build(){
? ? ? ? ? ? ????return new Student(this);
? ? ? ? }
? ? }
}
使用時(shí):
? new Student.Builder("小明", "26")
? ? ? ? ? ? ? ? .phone("183***")
? ? ? ? ? ? ? ? .address("西湖區(qū)")
? ? ? ? ? ? ? ? .build();
我們來看上面這個(gè)Student類
這個(gè)Student類的構(gòu)造方法設(shè)置成了private,說明調(diào)用者不能直接創(chuàng)建Student對(duì)象殃饿。Student類中有一個(gè)靜態(tài)內(nèi)部類Builder谋作,Student類有四個(gè)屬性,我們都設(shè)置成了final乎芳,只提供getter方法遵蚜。并且在構(gòu)造方法中對(duì)他們進(jìn)行了初始化。構(gòu)造方法中只有一個(gè)參數(shù)就是Builder對(duì)象奈惑,我們Student對(duì)象的創(chuàng)建就在Builder中的build方法中吭净。所以總的來說我們是使用Builder 對(duì)象來創(chuàng)建Student對(duì)象。
接下來看這個(gè)靜態(tài)內(nèi)部類Builder
Builder類中的屬性和Student是一致的肴甸,有幾個(gè)方法很明顯是分別設(shè)置幾個(gè)屬性的寂殉,這幾個(gè)方法的返回值都是Builder本身,這是為了鏈?zhǔn)秸{(diào)用原在。Builder類的構(gòu)造方法中有兩個(gè)參數(shù)友扰,說明有兩個(gè)屬性是必傳的。其他的參數(shù)就可以使用鏈?zhǔn)秸{(diào)用按需設(shè)置了晤斩。當(dāng)屬性設(shè)置完畢之后就是調(diào)用build方法了焕檬,在build方法中創(chuàng)建了Student對(duì)象。這時(shí)我們要?jiǎng)?chuàng)建Student對(duì)象的目的也就完成了澳泵。
這樣來創(chuàng)建對(duì)象不僅可讀性強(qiáng)实愚,而且build()方法可以明確的告訴別人對(duì)象已經(jīng)創(chuàng)建完畢。特別優(yōu)雅兔辅。
對(duì)于這種有可選屬性的對(duì)象來說用Builder模式來創(chuàng)建對(duì)象有很明顯的優(yōu)勢腊敲。下面是兩種傳統(tǒng)寫法:
1. 需要寫很多參數(shù)不同的構(gòu)造方法,想想就惡心维苔。
2. 先利用無參構(gòu)造方法創(chuàng)建一個(gè)對(duì)象碰辅,再使用setter()方法把需要的參數(shù)set進(jìn)去。這樣也有明顯的缺點(diǎn),一個(gè)是對(duì)象創(chuàng)建沒有一個(gè)明顯的結(jié)束標(biāo)志介时,有可能會(huì)在對(duì)象需要的屬性還沒完全set進(jìn)去之前就被調(diào)用没宾,這就會(huì)出現(xiàn)問題了。另一個(gè)是這樣創(chuàng)建的這個(gè)對(duì)象就是可變的了沸柔,也就是說在我們創(chuàng)建好的基礎(chǔ)上繼續(xù)改變它循衰,給它set新的屬性或者刪除某個(gè)已經(jīng)set的屬性等。
Builder線程安全問題
Builder是非線程安全的褐澎,所以要對(duì)參數(shù)做合法性驗(yàn)證的話必需要在對(duì)象創(chuàng)建完成之后再檢查会钝。
正確的代碼示例是:
? public Student build(){
? ? ? ? ? ? Student stu=new Student(this);
? ? ? ? ? ? if(stu.getName().equals("二狗")){
? ? ? ? ? ? ? ? ????throw new IllegalStateException("這里不允許叫二狗");
? ? ? ? ? ? }
? ? ? ? ? ? ????return stu;
? ? ? ? }
一定要先創(chuàng)建對(duì)象再進(jìn)行參數(shù)驗(yàn)證,因?yàn)槿绻闰?yàn)證后創(chuàng)建對(duì)象的話工三,創(chuàng)建的對(duì)象的時(shí)候?qū)ο蟮膶傩钥赡芤呀?jīng)被其他線程改變了迁酸。