其實(shí)《高效Java》的第2章我讀了很多次,每一次都覺得很煩,因?yàn)榈?章說了很多小點(diǎn),可是只有兩個目的:
(1)不要重復(fù)創(chuàng)建對象
(2)消除過期對象的引用
弄懂這兩個點(diǎn),我覺得第2張理解的差不多了吧。我們看看怎么實(shí)現(xiàn)這兩點(diǎn)。
不要重復(fù)創(chuàng)建對象
(1)使用靜態(tài)工廠方法
調(diào)用靜態(tài)工廠方法,不用我們創(chuàng)建對象,直接調(diào)用方法。如String.valueOf()。但是并不是說,如果這么好用,方法能定義為靜態(tài)方法盡量定義咯。你傻呀,是有限制的好嗎?既然他是靜態(tài)方法,就使用使用靜態(tài)變量,不能使用對象變量。我們一般在哪里會用到呢?比如工具類啦,又比如獲取單例對象的時候啦。既然在這里說到了單例,接下來看看唄!
(2)使用單例
在《高效Java》中,提到使用“私有構(gòu)造函數(shù)強(qiáng)化不可實(shí)例化能力”,這也就是單例的思想啦。單例模式確保一個一個類只有一個實(shí)例,這個也可以避免我們重復(fù)創(chuàng)建對象,節(jié)省資源。但是不要什么類都使用單例喲?一般都是需要某個類只能實(shí)例化一個實(shí)例,避免多個實(shí)例引起程序邏輯錯誤的場合下使用。結(jié)合我自己的經(jīng)驗(yàn),單例還能聊一下。單例還分饑餓式和懶惰式。饑餓式呢就是在定義單例屬性的時候就初始化,缺點(diǎn)是可能會造成浪費(fèi)內(nèi)存,因?yàn)槟悴灰欢ㄊ褂盟?,但是他已?jīng)初始化了;優(yōu)點(diǎn)呢,就是保證線程安全,在多線程環(huán)境中,不會出現(xiàn)2個這樣的實(shí)例。懶惰式呢,就是在外部類要得到這個單例的時候,才初始化。由于不會一開始就初始化,所以不會造成內(nèi)存浪費(fèi);缺點(diǎn)呢,你要做好多線程安全處理。具體自己百度去吧,有很多解決方案。
(3)適當(dāng)?shù)膎ew,重用對象
在《高效Java》提到一個例子:
String s = new String("silly"); // Don't do this
String s = "No longer sliiy"; // Should do this
這樣只是使用了一個String實(shí)例,而不是創(chuàng)建一個新的實(shí)例。聽我說,String這個類比較特殊,JVM在處理字符串的時候會采用緩存的策略,也就是說,如果一個String的值在編譯期間就可以確定,那么系統(tǒng)會將這個字符串存儲到常量池當(dāng)中,下次如果有其他的String對象的值和之前出現(xiàn)的值相等,那么系統(tǒng)會自動將這個對象指向常量池里對應(yīng)的字符串,從而節(jié)約空間。這也是String的高效用法。其實(shí)這個是重用的思想,還有我們ListView,如果要做到節(jié)省內(nèi)存,就可以用到ViewHolder。啥是ViewHolder,問百度。消除過期對象的引用
因?yàn)镴ava有GC幫我們管理著內(nèi)存,所以我們new完一個對象,也就不用管它死活了。其實(shí)不然。如果我們new完一個對象,如果實(shí)例沒有引用了到它,那它就屬于一個GC可到達(dá)的地方,GC就可以清理他??墒怯龅竭@種情況:
class StackA {
public String[] ss = new String[10];
public int size = 0;
public void push(String s) {
if(size < 10) {
ss[size++] = s;
}
}
public String pop() {
if(size == 0) return;
return ss[size--]; // 內(nèi)存泄露的地方
}
}
我們看到泄露的地方,雖然size是--了,但是ss[size]還是存在的,數(shù)組還引用到它,但是并不使用,這就相當(dāng)于ss[size]已經(jīng)是過期的對象了,我們就應(yīng)該把他消除,怎么消除?
我們重寫這個pop這個方法,代碼如下:
public String pop() {
if(size == 0) return;
ss[size] = null;
return ss[size--];
}
就這么簡單我們就消除了過期的引用。因?yàn)檫@是一個經(jīng)典的例子,所以大家都應(yīng)該了解一下過期引用是個什么東東,應(yīng)該怎么消除。主要的思想還是,如果你不用了它(對象),就不要“占有”它(對象)。這樣就很好可以消除過期引用了。
還有一種就是使用軟引用和弱引用技術(shù),使用軟引用保存的對象,也可以成為GC回收的對象。至Android 2.3之后,GC更傾向于回收軟引用和弱引用,使用的時候注意判斷null。關(guān)于軟引用和弱引用的使用,大家還是百度吧!!
以上是個人總結(jié)出來的,如有不對,歡迎指正?。?!