Java 四種引用類型,強引用、軟引用、弱引用、虛引用

雲棲號資訊:【點擊查看更多行業資訊
在這裏您可以找到不同行業的第一手的上雲資訊,還在等什麼,快來!


前言

每種編程語言都有自己操作內存中元素的方式,例如在 C 和 C++ 裏是通過指針,對於指針的使用,大家肯定記得剛學習時候的痛苦。

作爲 OOP 面向對象編程的翹楚,在 Java 中一切都被視爲了對象。但其實操作時候的標識符並不是真正的對象,而是對象的一個引用(reference)。

通過將這個叫“引用”的標識符指向某個對象,之後便可以使用這個引用來實現操作對象了。

在 JDK1.2 之前,Java中的定義很傳統:如果 reference 類型的數據中存儲的數值代表的是另外一塊內存的起始地址,就稱爲這塊內存代表着一個引用。

Java 中的垃圾回收機制在判斷是否回收某個對象的時候,都需要依據“引用”這個概念。不同垃圾回收算法中,對引用的判斷方式有所不同,典型的有引用計數法和可達性分析法。

JDK1.2 之前,一個對象只有“已被引用”和"未被引用"兩種狀態,這將無法描述某些特殊情況下的對象,比如,當內存充足時需要保留,而內存緊張時才需要被拋棄的一類對象。

所以在 JDK.1.2 之後,Java 對引用的概念進行了擴充,將引用分爲了:強引用(Strong Reference)、軟引用(Soft Reference)、弱引用(Weak Reference)、虛引用(Phantom Reference)4 種,這 4 種引用的強度依次減弱。

本篇文章將對這四種引用類型進行詳細介紹。

正文

強引用

在 Java 中,我們默認聲明的時候,使用的是強引用,比如:

A a = new A();

只要使用了強引用,垃圾回收器將永遠不會回收被引用的對象。由於內存大小是一定的,當內存不足時,JVM 會直接拋出 OutOfMemoryError,因爲沒辦法回收去釋放空間。

如果想中斷強引用與對象之間的關係,可以顯示將其賦值爲 null,這樣一來JVM 就可以在適當的時候進行垃圾回收了。

A a = null;

軟引用

軟引用比強引用的程度減弱一些,表示是一些非必需但仍有用的對象。它的表現形式如下:

(1)內存足夠的時候,軟引用對象不會被回收。

(2)內存不足時,系統則會回收軟引用對象。

如果軟引用對象被回收之後,仍然沒有足夠的內存空間,然後會拋出內存溢出異常。

由於軟引用的這種特性,非常適合實現緩存技術,比如網頁緩存,圖片緩存等等。

在 JDK1.2 之後,用java.lang.ref.SoftReference類來表示軟引用。

弱引用

弱引用的強度還要更弱一些,它最主要的特點是:無論內存是否足夠,只要 JVM 開始進行垃圾回收,那些被弱引用關聯的對象都會被回收。

在 JDK1.2 之後,用 java.lang.ref.WeakReference 來表示弱引用。

我們可以用下面的小demo來測試一下:

private static void testWeakReference() {

        for (int i = 0; i < 10; i++) {
            byte[] buff = new byte[1024 * 1024];
            WeakReference<byte[]> sr = new WeakReference<>(buff);
            list.add(sr);
        }

        System.gc(); //主動通知垃圾回收
        for(int i=0; i < list.size(); i++){
            Object obj = ((WeakReference) list.get(i)).get();
            System.out.println(obj);
        }

    }

運行後會發現所有被弱引用關聯的對象都被垃圾回收了。

在學習弱引用的時候,一個經典的例子就是 ThreadLocal,通過源代碼我們發現它的內部數據結構就是由弱引用來實現的。

 static class ThreadLocalMap {

        /**
         * The entries in this hash map extend WeakReference, using
         * its main ref field as the key (which is always a
         * ThreadLocal object).  Note that null keys (i.e. entry.get()
         * == null) mean that the key is no longer referenced, so the
         * entry can be expunged from table.  Such entries are referred to
         * as "stale entries" in the code that follows.
         */
        static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }
}

在使用 ThreadLocal 時如果忘記了remove操作,很容易出現內存泄露的問題,後續有時間我會專門寫一篇文章再講講這個。

虛引用

虛引用是最弱的一種引用關係,如果一個對象僅持有虛引用,那麼它就和沒有任何引用一樣,它隨時可能會被回收。

在 JDK1.2 之後,用 PhantomReference 類來表示,通過查看這個類的源碼,發現它只有一個構造函數和一個 get() 方法,而且它的 get() 方法僅僅是返回一個 null,也就是說將永遠無法通過虛引用來獲取對象,虛引用必須要和 ReferenceQueue 引用隊列一起使用。

public class PhantomReference<T> extends Reference<T> {

    /**
     * Returns this reference object's referent.  Because the referent of a
     * phantom reference is always inaccessible, this method always returns
     * <code>null</code>.
     *
     * @return  <code>null</code>
     */

    public T get() {
        return null;
    }

    public PhantomReference(T referent, ReferenceQueue<? super T> q) {
        super(referent, q);
    }

}

總結

通過這篇文章介紹了 Java 語言中的四種引用類型,看完後我們發現這主要與Java 的垃圾回收機制和內存管理息息相關。JVM 作爲基礎,還是希望大家能夠紮實的掌握。

【雲棲號在線課堂】每天都有產品技術專家分享!
課程地址:https://yqh.aliyun.com/live

立即加入社羣,與專家面對面,及時瞭解課程最新動態!
【雲棲號在線課堂 社羣】https://c.tb.cn/F3.Z8gvnK

原文發佈時間:2020-08-03
本文作者:程序員大帝
本文來自:“掘金”,瞭解相關信息可以關注“掘金”

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