學(xué)習(xí)目的:
- 何時(shí)以及如何創(chuàng)建對(duì)象
- 何時(shí)以及如何避免創(chuàng)建對(duì)象
- 如何確保對(duì)象適時(shí)銷毀
- 如何管理對(duì)象銷毀前必須進(jìn)行的各種清理動(dòng)作
1. 考慮用靜態(tài)工廠方法代替構(gòu)造器
該靜態(tài)工廠方法不同于設(shè)計(jì)模式中的工廠方法衰猛,主要用于如何鮮明提供可定制化的實(shí)例構(gòu)造方法稀火。
優(yōu)勢(shì):
1. 靜態(tài)方法可以提供名稱唇敞,可以確切地描述返回的對(duì)象恨统,更易于代碼閱讀、理解。
2. 不必每次調(diào)用的時(shí)候都創(chuàng)建新的對(duì)象,這樣就可以反復(fù)使用預(yù)先構(gòu)建好的對(duì)象碘橘,或者將創(chuàng)建的對(duì)象緩存起來(lái),有利于提升性能吱肌,這種方式在單例模式中應(yīng)用較多痘拆。
3. 通過(guò)依賴于接口,而不依賴與具體實(shí)現(xiàn)岩榆,可以靈活返回實(shí)現(xiàn)該接口的子類错负。
4. 構(gòu)造參數(shù)化實(shí)例時(shí)坟瓢,代碼更加簡(jiǎn)潔。
劣勢(shì):
要提供可在外部實(shí)例化的類犹撒,否則優(yōu)勢(shì)3無(wú)法體現(xiàn)折联,而已靜態(tài)工廠方法依然只是一個(gè)普通的靜態(tài)方法,需要提供注釋來(lái)表示她的構(gòu)造功能說(shuō)明识颊。
針對(duì)劣勢(shì)方面诚镰,可以提供通用的方法名稱如valueOf,newInstance祥款,getInstance等來(lái)標(biāo)示該方法的存在意義清笨。更主要的是在構(gòu)建對(duì)象時(shí)考慮需不需要通過(guò)靜態(tài)工廠來(lái)靈活構(gòu)造實(shí)例對(duì)象。
2. 遇到多個(gè)構(gòu)造器參數(shù)時(shí)要考慮用Builder
在實(shí)例化某些對(duì)象時(shí)刃跛,可能需要提供多個(gè)參數(shù)抠艾,而有些參數(shù)是必須的或可選的,這個(gè)時(shí)候一般就會(huì)提供多個(gè)構(gòu)造器來(lái)實(shí)例化不同的對(duì)象桨昙。
- 對(duì)象的實(shí)例化參數(shù)增多检号,意味著構(gòu)造方法也同時(shí)增多,代碼的可讀性也隨之下降蛙酪。
- 采取builder模式齐苛,完全由Builder來(lái)靈活定制對(duì)象的實(shí)例屬性,針對(duì)每個(gè)屬性提供一個(gè)set方法即可桂塞。
- 調(diào)用build方法來(lái)構(gòu)建對(duì)象凹蜂,并在對(duì)象構(gòu)建方法中即可一次性檢測(cè)參數(shù)設(shè)置異常
3. 用私有構(gòu)造器或枚舉來(lái)強(qiáng)化Singleton模式
- 對(duì)于實(shí)現(xiàn)Singleton模式一般都需要將構(gòu)造器私有化,但通過(guò)Java的反射方式依然能破壞該模式阁危,可以通過(guò)修改私有構(gòu)造器在實(shí)例化第二個(gè)對(duì)象時(shí)直接拋出異常來(lái)保證單例不被破壞玛痊。
- 對(duì)象的序列化和反序列號(hào)也能破壞單例模式,需要提供readResolve方法來(lái)保證返回實(shí)例相同狂打。
- 單元素枚舉方式來(lái)保證Singleton模式卿啡,自動(dòng)提供上述兩種功能,并且能保證線程安全菱父,是目前Singleton最佳方法。
4. 通過(guò)私有構(gòu)造器來(lái)強(qiáng)化不可實(shí)例化的能力
有時(shí)候需要編寫(xiě)一個(gè)只需要包含靜態(tài)方法或靜態(tài)域的類剑逃,用于提供一些工具性的方法浙宜,而這樣的工具類一般不希望被實(shí)例化。
在編寫(xiě)類時(shí)若不提供構(gòu)造方法蛹磺,則系統(tǒng)會(huì)自動(dòng)生成一個(gè)無(wú)參構(gòu)造器粟瞬,所以我們需要編寫(xiě)一個(gè)私有構(gòu)造器,并在內(nèi)部直接拋出異常萤捆,你也可以在上面編寫(xiě)注釋表明意圖裙品。
5. 避免創(chuàng)建不必要的對(duì)象
what:
不必要的對(duì)象的鮮明特征就是對(duì)于反復(fù)使用的某一功能或方法俗批,在每次運(yùn)行時(shí)都會(huì)重新創(chuàng)建對(duì)象,而不是使用之前已有對(duì)象市怎。
why:
對(duì)象的創(chuàng)建是需要代價(jià)的岁忘,大量創(chuàng)建無(wú)用對(duì)象既增加占用內(nèi)存也減少了代碼運(yùn)行流暢,比如創(chuàng)建數(shù)據(jù)庫(kù)連接對(duì)象需要較大開(kāi)銷区匠,如果能復(fù)用之前已有連接對(duì)象干像,對(duì)于提升性能具有顯著幫助。
how:
可以考慮在將常用對(duì)象緩存住驰弄,比如靜態(tài)域等等麻汰。但是不要總是考慮緩存對(duì)象這樣不一定能提升能力,只是對(duì)于一些創(chuàng)建對(duì)象開(kāi)銷較大并且使用較多對(duì)象緩存戚篙,創(chuàng)建一些微對(duì)象對(duì)于代碼運(yùn)行并沒(méi)有多少影響五鲫。
6. 消除過(guò)期的對(duì)象引用
Java是能夠自動(dòng)進(jìn)行垃圾回收但前提是對(duì)象沒(méi)有被引用,而在代碼編寫(xiě)過(guò)程中由于緩存或一些其他使用導(dǎo)致在不需要該對(duì)象時(shí)岔擂,依然內(nèi)部持有該對(duì)象的引用位喂,導(dǎo)致該對(duì)象沒(méi)有被及時(shí)釋放,從而導(dǎo)致內(nèi)存泄漏智亮。所以對(duì)不再需要使用的對(duì)象盡量置為null或者對(duì)于一些對(duì)象采用弱引用忆某。
7. 避免使用終結(jié)方法
首先終結(jié)方法是JVM自己調(diào)用的,而隨著垃圾回收算法改變阔蛉,終結(jié)算法調(diào)用時(shí)刻也是不一致的弃舒,所以不要依賴終結(jié)方法做一些額外的操作,這樣會(huì)導(dǎo)致整個(gè)應(yīng)用崩潰的状原,而且Java并不提供能直接命令系統(tǒng)進(jìn)行垃圾回收的方法聋呢,System.gc()也只是建議開(kāi)始系統(tǒng)回收的。
如果顯示覆蓋終結(jié)方法颠区,來(lái)一定要調(diào)用父類終結(jié)方法削锰,因?yàn)镴VM不會(huì)主動(dòng)調(diào)用父類的gc方法的。