10分鐘帶你理解Java中的弱引用_java

來源:互聯網
上載者:User

前言

本文嘗試從What、Why、How這三個角度來探索Java中的弱引用,協助大家理解Java中弱引用的定義、基本使用情境和使用方法。

一、 What——什麼是弱引用?

Java中的弱引用具體指的是java.lang.ref.WeakReference<T>類,我們首先來看一下官方文檔對它做的說明:

     弱引用對象的存在不會阻止它所指向的對象被記憶體回收行程回收。弱引用最常見的用途是實現規範映射(canonicalizing mappings,比如雜湊表)。

     假設垃圾收集器在某個時間點決定一個對象是弱可達的(weakly reachable)(也就是說當前指向它的全都是弱引用),這時垃圾收集器會清除所有指向該對象的弱引用,然後把這個弱可達對象標記為可終結(finalizable)的,這樣它隨後就會被回收。與此同時或稍後,垃圾收集器會把那些剛清除的弱引用放入建立弱引用對象時所指定的引用隊列(Reference Queue)中。

實際上,Java中存在四種引用,它們由強到弱依次是:強引用軟引用弱引用虛引用

下面我們簡單介紹下除弱引用外的其他三種引用:

1、強引用(Strong Reference):通常我們通過new來建立一個新對象時返回的引用就是一個強引用,若一個對象通過一系列強引用可到達,它就是強可達的(strongly reachable),那麼它就不被回收

2、軟引用(Soft Reference):軟引用和弱引用的區別在於,若一個對象是弱引用可達,無論當前記憶體是否充足它都會被回收,而軟引用可達的對象在記憶體不充足時才會被回收,因此軟引用要比弱引用“強”一些

3、虛引用(Phantom Reference):虛引用是Java中最弱的引用,那麼它弱到什麼程度呢?它是如此脆弱以至於我們通過虛引用甚至無法擷取到被引用的對象,虛引用存在的唯一作用就是當它指向的對象被回收後,虛引用本身會被加入到引用隊列中,用作記錄它指向的對象已被回收。

二、Why——為什麼使用弱引用?

考慮下面的情境:現在有一個Product類代表一種產品,這個類被設計為不可擴充的,而此時我們想要為每個產品增加一個編號。一種解決方案是使用HashMap<Product, Integer> 。於是問題來了,如果我們已經不再需要一個Product對象存在於記憶體中(比如已經賣出了這件產品),假設指向它的引用為productA,我們這時會給productA賦值為null,然而這時productA過去指向的Product對象並不會被回收,因為它顯然還被HashMap引用著。所以這種情況下,我們想要真正的回收一個Product對象,僅僅把它的強引用賦值為null是不夠的,還要把相應的條目從HashMap中移除。顯然“從HashMap中移除不再需要的條目”這個工作我們不想自己完成,我們希望告訴垃圾收集器:在只有HashMap中的key在引用著Product對象的情況下,就可以回收相應Product對象了。顯然,根據前面弱引用的定義,使用弱引用能協助我們達成這個目的。我們只需要用一個指向Product對象的弱引用對象來作為HashMap中的key就可以了。

三、How——如何使用弱引用?

拿上面介紹的情境舉例,我們使用一個指向Product對象的弱引用對象來作為HashMapkey,只需這樣定義這個弱引用對象:

Product productA = new Product(...);WeakReference<Product> weakProductA = new WeakReference<>(productA);

現在,若引用對象weakProductA就指向了Product對象productA。那麼我們怎麼通過weakProduct擷取它所指向的Product對象productA呢?

很簡單,只需要下面這句代碼:

Product product = weakProductA.get();

實際上,對於這種情況,Java類庫為我們提供了WeakHashMap類,使用和這個類,它的鍵自然就是弱引用對象,無需我們再手動封裝原始對象。這樣一來,當productA變為null時(表明它所引用的Product已經無需存在於記憶體中),這時指向這個Product對象的就是由弱引用對象weakProductA了,那麼顯然這時候相應的Product對象時弱可達的,所以指向它的弱引用會被清除,這個Product對象隨即會被回收,指向它的弱引用對象會進入引用隊列中。

四、引用隊列

下面我們來簡單地介紹下引用隊列的概念。實際上,WeakReference類有兩個建構函式:

//建立一個指向給定對象的弱引用WeakReference(T referent) //建立一個指向給定對象並且登記到給定引用隊列的弱引用WeakReference(T referent, ReferenceQueue<? super T> q)

我們可以看到第二個構造方法中提供了一個ReferenceQueue類型的參數,通過提供這個參數,我們便把建立的弱引用對象註冊到了一個引用隊列上,這樣當它被記憶體回收行程清除時,就會把它送入這個引用隊列中,我們便可以對這些被清除的弱引用對象進行統一管理。

五、總結

好了,這篇文章的內容到這就結束了,由於個人水平有限,敘述中難免存在不準確或是不清晰的地方,希望大家可以指出,謝謝大家對雲棲社區的支援。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.