JAVA虛擬機(JVM)--伍(四種引用)

JAVA 四中引用類型
 
1、強引用(StrongReference)
在 Java 中最常見的就是強引用,把一個對象賦給一個引用變量,這個引用變量就是一個強引用。當一個對象被強引用變量引用時,它處於可達狀態,它是不可能被垃圾回收機制回收的,即使該對象以後永遠都不會被用到 JVM 也不會回收。因此強引用是造成 Java 內存泄漏的主要原因之一。
例如:
Object obj = new Object();

2、軟引用(SoftReference)

軟引用需要用 SoftReference 類來實現,對於只有軟引用的對象來說,當系統內存足夠時它不會被回收,當系統內存空間不足時它會被回收。軟引用通常用在對內存敏感的程序中。
軟引用可用來實現內存敏感的高速緩存。
Object obj = new Object();  //強引用
ReferenceQueue<Object> referenceQueuee = new ReferenceQueue<>(); //引用隊列
// 軟引用和引用隊列聯合使用,如果軟引用所引用的對象被垃圾回收器回收,JVM就會把這個軟引用加入到引用隊列中
SoftReference softReference = new SoftReference(str, referenceQueuee); 

當內存不足時,等價於:

if(JVM.內存不足()){
    obj = null;
    System.gc();
}

3、弱引用(WeakReference)

弱引用需要用 WeakReference 類來實現,它比軟引用的生存期更短,對於只有弱引用的對象來說,只要垃圾回收機制一運行,不管 JVM 的內存空間是否足夠,總會回收該對象佔用的內存。
@Test
public void testWeakReference() throws InterruptedException {
    ReferenceQueue<Object> referenceQueuee = new ReferenceQueue<>();
    Object weakObject = new Object();
    //弱引用
    WeakReference weakReference = new WeakReference(weakObject, referenceQueuee);
    System.out.println("WeakReference:" + weakReference.get());
    System.out.println("referenceQueuee:" + referenceQueuee.poll());

    weakObject = null;
    System.gc();
    Thread.sleep(2000);
    System.out.println("WeakReference:" + weakReference.get());
    System.out.println("referenceQueuee:" + referenceQueuee.poll());
}

測試結果:

WeakReference:java.lang.Object@694f9431
referenceQueuee:null
WeakReference:null
referenceQueuee:java.lang.ref.WeakReference@f2a0b8e

4、虛引用(PhantomReference)

虛引用需要 PhantomReference 類來實現,它不能單獨使用,必須和引用隊列聯合使用。虛引用的主要作用是跟蹤對象被垃圾回收的狀態。
@Test
public void testPhantomReference() throws InterruptedException {
    //虛引用:功能,不會影響到對象的生命週期的,
   // 但是能讓程序員知道該對象什麼時候被 回收了
   ReferenceQueue<Object> referenceQueuee = new ReferenceQueue<>();
   Object phantomObject = new Object();
   PhantomReference phantomReference = new PhantomReference(phantomObject,referenceQueuee);
   phantomObject = null;
   System.out.println("phantomObject:" + phantomObject);//null
   System.out.println("phantomReference" + referenceQueuee.poll());//null
   
   System.gc();
   Thread.sleep(2000);
   System.out.println("referenceQueuee:" + referenceQueuee.poll());
}

測試結果:

phantomObject:null
phantomReferencenull
referenceQueuee:java.lang.ref.PhantomReference@694f9431

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章