銷毀實(shí)例時(shí)注意事項(xiàng)
內(nèi)存泄露
出現(xiàn)內(nèi)存泄露的3種常見情況及應(yīng)對(duì)原則
- 類中存在自我管理內(nèi)存現(xiàn)象
類中存在容器(如數(shù)組)維持其他對(duì)象罐农,凡是存在與該數(shù)組中的對(duì)象都會(huì)被強(qiáng)制持有,垃圾回收器不會(huì)對(duì)其進(jìn)行回收,如果該容器中存在已過期對(duì)象時(shí),這就導(dǎo)致了內(nèi)存泄漏。我們應(yīng)該強(qiáng)制將該容器的該元素置為空,讓該過期對(duì)象失去引用抑堡,進(jìn)入垃圾回收器的視野中
- 緩存技術(shù)的使用
我們建立緩存機(jī)制的時(shí)候,要有合理的以時(shí)間锁蠕、活躍度為排序原則的容器夷野,并且設(shè)立緩存容器的大小,但達(dá)到回收條件時(shí)(大小/數(shù)量達(dá)到最大值)荣倾,首先按時(shí)間悯搔、活躍度排序定義回收優(yōu)先級(jí)。
- 監(jiān)聽器或者回調(diào)的使用
因?yàn)闃I(yè)務(wù)時(shí)序的非正常進(jìn)行導(dǎo)致監(jiān)聽/回調(diào)對(duì)象的未取消注冊(cè),此時(shí)被注冊(cè)對(duì)象依然持有該listener妒貌,但注冊(cè)者已銷毀或者已不關(guān)注該回調(diào)通危。最優(yōu)的做法就是保持該對(duì)象以弱引用的方式被使用,一旦其他顯示引用不存在灌曙,該弱引用會(huì)進(jìn)入垃圾回收器的視野中
終結(jié)方法(finalizer)
終結(jié)方法指的就是finalize()
方法
- 工作原理
當(dāng)GC檢測到對(duì)象已廢棄后菊碟,會(huì)檢測對(duì)象的
finalize()
方法是否被復(fù)寫,未復(fù)寫進(jìn)行正吃诖蹋回收逆害,否則將該對(duì)象加入一個(gè)隊(duì)列,通過一個(gè)低優(yōu)先級(jí)線程輪詢執(zhí)行每個(gè)對(duì)象的finalize()
方法蚣驼,執(zhí)行過的對(duì)象被移除出該隊(duì)列魄幕,GC會(huì)再次檢測對(duì)象是否有效,無效執(zhí)行回收颖杏,有效則“復(fù)活”
- 不可預(yù)期
該方法不能保證能得到及時(shí)執(zhí)行纯陨,甚至不能保證被執(zhí)行,這會(huì)導(dǎo)致依賴該方法的資源回收操作變得不可預(yù)期(如文件操作的描述符回收)留储,并且背壓(銷毀對(duì)象的速度趕不上待銷毀對(duì)象加入垃圾回收器的速度)的存在翼抠,加劇了這一現(xiàn)象,由于終結(jié)方法的執(zhí)行線程低優(yōu)先級(jí)获讳,可能你的終結(jié)方法一直得不到執(zhí)行
- 可移植性差
執(zhí)行終結(jié)方法是GC的一個(gè)重要特性阴颖,但GC在不同的JVM上的規(guī)則是不一樣的,所以依賴該方法的操作會(huì)導(dǎo)致程序在不同的JVM上表現(xiàn)可能大有不同赔嚎,甚至導(dǎo)致了業(yè)務(wù)邏輯的不統(tǒng)一膘盖,這無疑增加了移植的成本
- 終結(jié)方法拋出異常
通常情況下,未捕捉的異常會(huì)終止線程尤误,并打出堆棧信息。但如果該異常發(fā)生在終結(jié)方法的執(zhí)行過程中结缚,連警告信息都不會(huì)打出來损晤,很難進(jìn)行debug
- 性能損耗
《effective java》一書中作者舉例,正常銷毀一個(gè)對(duì)象耗時(shí)5.6ns红竭,而增加了終結(jié)方法后尤勋,耗時(shí)增加到2400ns(未驗(yàn)證)
- 顯式定義的業(yè)務(wù)終止方法+ try-finally
通過在finally語句塊中,調(diào)用自定義的顯式的終止方法釋放資源茵宪,這是最經(jīng)濟(jì)有效的方式最冰,而不是直接通過終結(jié)方法釋放資源
- 終止方法+終結(jié)方法
終結(jié)方法可以做沒有調(diào)用顯式終止方法時(shí)的第二層防護(hù)(safety net),畢竟延遲釋放總比不釋放好稀火。另外如果到達(dá)了第二層終結(jié)方法出才進(jìn)行資源釋放暖哨,證明程序存在內(nèi)存泄漏的BUG,應(yīng)及時(shí)修復(fù)
- 本地對(duì)等體
將java世界的對(duì)象通過
JNI
賦值給native世界的對(duì)象凰狞,該native對(duì)象無法被GC回收篇裁,如果在沒有顯式的JNI方法能做這個(gè)回收動(dòng)作的情況下沛慢,終結(jié)方法便是最好的選擇。
- 復(fù)寫時(shí)要注意
復(fù)寫父類的
finalize()
方法的時(shí)候达布,一定要在finally
子句中調(diào)用super.finalize()
团甲,否則父類的終結(jié)方法不會(huì)執(zhí)行。
總結(jié): 除了二次防護(hù)用法和終止本地對(duì)等體對(duì)象用法黍聂,請(qǐng)不要使用終結(jié)方法(finalize()
)