Java中有四種引用類型:StrongReference(強引用),SoftReference(軟引用),WeakReference(弱引用),PhantomReference(虛引用)。這四種引用的強度按照上面的順序依次減弱。強引用就是咱們一般使用的引用類型,若是在GC Roots引用鏈上有一個強引用可以到達一個對象,那麼這個對象就不會被垃圾回收。若是一個對象的引用爲軟引用,則垃圾收集器會視內存使用狀況來決定是否回收對象:若是內存足夠則不回收對象,否定回收。而弱引用則不相同:垃圾收集器無論內存是否足夠只要一個對象只有一個弱引用,就回收對象。虛引用不會決定對象的生命週期,垃圾收集器任什麼時候候均可以回收虛引用引用的對象。緩存
若是一個對象具備強引用,那垃圾回收器毫不會回收它。當內存空間不足,Java虛擬機會拋出OutOfMemoryError錯誤,使程序異常終止。 示例: String strongRef = new String("strong reference");
bash
若是一個對象只具備軟引用,則內存空間足夠,垃圾回收器就不會回收它;若是內存空間不足了,就會回收這些對象的內存。只要垃圾回收器沒有回收它,該對象就能夠被程序使用。軟引用可用來實現內存敏感的高速緩存。spa
軟引用能夠和一個引用隊列(ReferenceQueue)聯合使用,若是軟引用所引用的對象被垃圾回收器回收,Java虛擬機就會把這個軟引用加入到與之關聯的引用隊列中。 示例:線程
// 示例1
SoftReference<String> softRef = new SoftReference<String>(new String("soft reference"));
// 示例2
ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();
SoftReference<String> softRef = new SoftReference<String>(new String("soft reference"), referenceQueue);
複製代碼
弱引用與軟引用的區別在於:只具備弱引用的對象擁有更短暫的生命週期。在垃圾回收器線程掃描它所管轄的內存區域的過程當中,一旦發現了只具備弱引用的對象,無論當前內存空間足夠與否,都會回收它的內存。不過,因爲垃圾回收器是一個優先級很低的線程,所以不必定會很快發現那些只具備弱引用的對象。code
弱引用能夠和一個引用隊列(ReferenceQueue)聯合使用,若是弱引用所引用的對象被垃圾回收,Java虛擬機就會把這個弱引用加入到與之關聯的引用隊列中。 實例:對象
// 示例1
WeakReference<String> weakRef = new WeakReference<String>(new String("hello ref"));
// 示例2
ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();
WeakReference<String> weakRef = new WeakReference<String>(new String("weak reference"), referenceQueue);
複製代碼
「虛引用」顧名思義,就是形同虛設,與其餘幾種引用都不一樣,虛引用並不會決定對象的生命週期。若是一個對象僅持有虛引用,那麼它就和沒有任何引用同樣,在任什麼時候候均可能被垃圾回收器回收。生命週期
虛引用主要用來跟蹤對象被垃圾回收器回收的活動。虛引用與軟引用和弱引用的一個區別在於:虛引用必須和引用隊列 (ReferenceQueue)聯合使用。當垃圾回收器準備回收一個對象時,若是發現它還有虛引用,就會在回收對象的內存以前,把這個虛引用加入到與之 關聯的引用隊列中。 示例:隊列
ReferenceQueue<String> referenceQueue = new ReferenceQueue<String>();
PhantomReference<String> referent = new PhantomReference<String>(new String("phantom reference"), referenceQueue);
複製代碼
引用類型 | 回收時機 | 是否可能致使內存泄露 |
---|---|---|
強引用 | 不回收 | 可能 |
軟引用 | 視內存狀況而定 | 不可能 |
弱引用 | 回收 | 不可能 |
虛引用 | 回收 | 可能 |