02 遇到多個構(gòu)造器參數(shù)時要考慮用構(gòu)建器

靜態(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ù)都是可選的時候就更完美了~斤彼。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市蘸泻,隨后出現(xiàn)的幾起案子琉苇,更是在濱河造成了極大的恐慌,老刑警劉巖悦施,帶你破解...
    沈念sama閱讀 218,386評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件并扇,死亡現(xiàn)場離奇詭異,居然都是意外死亡抡诞,警方通過查閱死者的電腦和手機穷蛹,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評論 3 394
  • 文/潘曉璐 我一進店門土陪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人肴熏,你說我怎么就攤上這事鬼雀。” “怎么了蛙吏?”我有些...
    開封第一講書人閱讀 164,704評論 0 353
  • 文/不壞的土叔 我叫張陵源哩,是天一觀的道長。 經(jīng)常有香客問我鸦做,道長励烦,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,702評論 1 294
  • 正文 為了忘掉前任馁龟,我火速辦了婚禮崩侠,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘坷檩。我一直安慰自己却音,他們只是感情好,可當我...
    茶點故事閱讀 67,716評論 6 392
  • 文/花漫 我一把揭開白布矢炼。 她就那樣靜靜地躺著系瓢,像睡著了一般。 火紅的嫁衣襯著肌膚如雪句灌。 梳的紋絲不亂的頭發(fā)上夷陋,一...
    開封第一講書人閱讀 51,573評論 1 305
  • 那天,我揣著相機與錄音胰锌,去河邊找鬼骗绕。 笑死,一個胖子當著我的面吹牛资昧,可吹牛的內(nèi)容都是我干的酬土。 我是一名探鬼主播,決...
    沈念sama閱讀 40,314評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼格带,長吁一口氣:“原來是場噩夢啊……” “哼撤缴!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起叽唱,我...
    開封第一講書人閱讀 39,230評論 0 276
  • 序言:老撾萬榮一對情侶失蹤屈呕,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后棺亭,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體虎眨,經(jīng)...
    沈念sama閱讀 45,680評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,873評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了专甩。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片钟鸵。...
    茶點故事閱讀 39,991評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖涤躲,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情贡未,我是刑警寧澤种樱,帶...
    沈念sama閱讀 35,706評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站俊卤,受9級特大地震影響嫩挤,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜消恍,卻給世界環(huán)境...
    茶點故事閱讀 41,329評論 3 330
  • 文/蒙蒙 一岂昭、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧狠怨,春花似錦约啊、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至憎蛤,卻和暖如春外傅,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背俩檬。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評論 1 270
  • 我被黑心中介騙來泰國打工萎胰, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人棚辽。 一個月前我還...
    沈念sama閱讀 48,158評論 3 370
  • 正文 我出身青樓技竟,卻偏偏與公主長得像,于是被迫代替她去往敵國和親晚胡。 傳聞我的和親對象是個殘疾皇子灵奖,可洞房花燭夜當晚...
    茶點故事閱讀 44,941評論 2 355

推薦閱讀更多精彩內(nèi)容