咱們知道在Java中除了基礎的數據類型之外,其它的都爲引用類型。 而Java根據其生命週期的長短將引用類型又分爲強引用、軟引用、弱引用、幻象引用 。面試
正常狀況下咱們平時基本上咱們只用到強引用類型,而其餘的引用類型咱們也就在面試中,或者平日閱讀類庫或其餘框架源碼的時候才能見到。緩存
咱們平日裏面的用到的new了一個對象就是強引用,例如 Object obj = new Object();
當JVM的內存空間不足時,寧願拋出OutOfMemoryError使得程序異常終止也不肯意回收具備強引用的存活着的對象!bash
記住是存活着,不多是你new一個對象就永遠不會被GC回收。當一個普通對象沒有其餘引用關係,只要超過了引用的做用域或者顯示的將引用賦值爲null時,你的對象就代表不是存活着,這樣就會能夠被GC回收了。固然回收的時間是不必定的具體得看GC回收策略。框架
軟引用的生命週期比強引用短一些。軟引用是經過SoftReference
類實現的。分佈式
Object obj = new Object();
SoftReference softObj = new SoftReference(obj);
obj = null; //去除強引用
複製代碼
這樣就是一個簡單的軟引用使用方法。經過get()
方法獲取對象。當JVM認爲內存空間不足時,就回去試圖回收軟引用指向的對象,也就是說在JVM拋出OutOfMemoryError
以前,會去清理軟引用對象。軟引用能夠與引用隊列(ReferenceQueue)
聯合使用。函數
Object obj = new Object();
ReferenceQueue queue = new ReferenceQueue();
SoftReference softObj = new SoftReference(obj,queue);
obj = null; //去除強引用
複製代碼
當softObj
軟引用的obj
被GC回收以後,softObj
對象就會被塞到queue
中,以後咱們能夠經過這個隊列的poll()
來檢查你關心的對象是否被回收了,若是隊列爲空,就返回一個null
。反之就返回軟引用對象也就是softObj
。性能
軟引用通常用來實現內存敏感的緩存,若是有空閒內存就能夠保留緩存,當內存不足時就清理掉,這樣就保證使用緩存的同時不會耗盡內存。例如圖片緩存框架中緩存圖片就是經過軟引用的。spa
弱引用是經過WeakReference
類實現的,它的生命週期比軟引用還要短,也是經過get()
方法獲取對象。code
Object obj = new Object();
WeakReference<Object> weakObj = new WeakReference<Object>(obj);
obj = null; //去除強引用
複製代碼
在GC的時候,無論內存空間足不足都會回收這個對象,一樣也能夠配合ReferenceQueue
使用,也一樣適用於內存敏感的緩存。ThreadLocal
中的key就用到了弱引用。對象
也稱虛引用,是經過PhantomReference
類實現的。任什麼時候候可能被GC回收,就像沒有引用同樣。
Object obj = new Object();
ReferenceQueue queue = new ReferenceQueue();
PhantomReference<Object> phantomObj = new PhantomReference<Object>(obj , queue);
obj = null; //去除強引用
複製代碼
沒法經過虛引用訪問對象的任何屬性或者函數。那就要問了要它有什麼用?虛引用僅僅只是提供了一種確保對象被finalize
之後來作某些事情的機制。好比說這個對象被回收以後發一個系統通知啊啥的。虛引用是必須配合ReferenceQueue
使用的,具體使用方法和上面提到軟引用的同樣。主要用來跟蹤對象被垃圾回收的活動。
若有錯誤歡迎指正!
我的公衆號:yes的練級攻略
有相關面試進階(分佈式、性能調優、經典書籍pdf)資料等待領取