Java有四大引用,讓開發者去管理對象的生命周期。強引用、軟引用、弱引用、軟引用。今天就讓我們簡單的了解一下吧。
強引用:當內存不足時,也不會被回收,所以強引用是造成內存泄漏的原因之一,當強引用對象不使用時,我們應該弱化他,從而使GC能夠回收。
軟引用:內存足夠時,不回收,但是當內存不足時,就會被回收。
弱引用:當垃圾回收器發現它時,它就會被回收。
虛引用:這是一個最虛幻的引用類型 . 無論是從哪里都無法再次返回被虛引用所引用的對象 . 虛引用在系統垃圾回收器開始回收對象時 , 將直接調用 finalize() 方法 , 但不會立即將其加入回收隊列 . 只有在真正對象被 GC 清除時 , 才會將其加入 Reference 隊列中去 .
當多次運行系統垃圾回收后,IBM JVM 將軟引用一并加入了回收隊列中,并運行了其 finalize 方法。另外,即使經過很多次系統垃圾回收,虛引用也沒有被加入到隊列中去。不知道這是不是 IBM JVM 的一個小小的 BUG 所在。
SoftReference 中 Oracle JVM 的表現滿足規范,只當內存不足時才進行回收。而 IBM JVM 的策略則更為積極,在內存尚且充足的情況下也進行了回收,值得注意。PhantomReference 中 Oracle JVM 的表現滿足規范,執行 finalize 后若干次 GC 就被添加到了 Queue 中。而 IBM JVM 則始終沒有被添加到 Queue 中導致了死循環。所以在使用 PhantomReference 時出現類似的情況時,可以考慮是否是因為使用了不同 JVM 所導致。
強引用
向我們平時new的對象都是強引用。
Object wangscaler=new Object();復制代碼
因為強引用不會被回收,如果我們的對象不再使用了,可以
wangscaler = null;復制代碼
將其弱化,等待垃圾回收時進行回收。
軟引用
軟引用一般用來做緩存。當內存充足時,將一些數據作為緩存,當內存不夠用時,進行回收。像圖片處理技術就會將用戶上傳的圖片加入緩存。
SoftReference wangscaler = new SoftReference<>(new Object());復制代碼
弱引用
WeakReference wangscaler = new WeakReference();復制代碼
弱引用和軟引用相似,同樣可以用來做緩存,不同的是弱引用的生命周期更短,每次GC的時候,都會被回收,所以應該用于調用頻率相對低一些的應用。
虛引用
ReferenceQueue wangscaler = new ReferenceQueue<>();
PhantomReference scaler = new PhantomReference<>(new Object(), wangscaler);復制代碼
必須和引用隊列一塊使用。我們如果有對象被回收時做一些特殊處理,可以采用虛引用。他就像是一種通知機制,和Spring的后置通知一樣,告訴我們對象已經被回收了。ThreadLocal底層代碼就使用虛引用。值得注意的是虛引用的get()的結果永遠都是null,無論有沒有被清理,都無法獲得到對象。
引用隊列
像軟引用、弱引用、虛引用三種引用都可以結合引用隊列ReferenceQueue一塊使用。當三種引用的對象被回收時,加入引用隊列,通過引用隊列可以了解JVM垃圾回收情況。
對比
總結
使用四種引用來決定對象的生命周期,就像我們日常的生活用品一樣,有些物品很快被扔進垃圾桶進行回收,有些物品當家里放不下了就扔出去了,而有些物品什么時候都不能扔,你的存折你扔扔試試。所以當內存不足時,會拋出OutOfMemory異常,也不會將強引用的對象回收。
虛引用有一個很重要的用途就是用來做堆外內存的釋放,DirectByteBuffer就是通過虛引用來實現堆外內存的釋放的。
更多關于“Java培訓”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓經驗,課程大綱更科學更專業,有針對零基礎的就業班,有針對想提升技術的好程序員班,高品質課程助理你實現java程序員夢想。