java中的Reference類型用法說明
本文簡要總結java中的Reference類型。
最近在研讀jdk并發框架,其中AQS是重點,由于我打破砂鍋問到底的輕微強迫癥,google了AQS作者Doug Lea的論文原文[The java.util.concurrent Synchronizer Framework],有興趣的同學可以自行下載。其中談到設計同步框架的核心是選擇一個嚴格意義上的FIFO隊列,作為阻塞線程隊列并對其進行維護。
對此主要由兩種選擇,一個是MCS鎖,另一個時CLH鎖。因為CLH鎖比MCS對取消和超時的處理更方便,所以AQS就選擇將CLH鎖作為基礎對其進行改進。于是我又打算先弄懂什么是CLH鎖,在網上搜索了一圈之后找到很多人對CLH鎖進行了java實現,實現中用到了ThreadLocal類型,于是我發現我好像對ThreadLocal也不太熟,于是去看openjdk的源碼,又發現ThreadLocal的內部類ThreadLocalMap中的Entry是繼承自WeakReference,好了,既然這幾個我好像都沒弄得很明白過,所以我決定先了解一下Reference。
在我們平時開發過程中很少會遇到需要與各種不同類型的reference打交道的時候,所以很多時候我們在自己寫的代碼中很少會碰到需要使用不同的reference類型,可能很多人也會向我這樣,想要深入學習一下jdk源碼或者其他某些框架的源碼的時候才會看到諸如WeakReference這樣的類型。
問題
假設在一個應用中,需要從一個名為test的數據庫表中獲取數據。但凡有點經驗的開發人員都會避免應用獲取相同的數據每次都去查詢數據庫,因為I/O操作過去頻繁勢必會降低應用性能。
顯然,我們首先想到的就是使用緩存。應用首先查詢緩存,如果需要的數據存在直接拿來用就好;如果緩存未命中,才去數據庫查詢,并且把查詢到的數據放入緩存,以便下次應用發起相同請求時可以直接從緩存獲取數據而不用再次去數據庫查詢。
使用緩存會提高性能嗎?
答案是這需要根據具體情況分析,如果從test獲取需要緩存的數據量較少,使用緩存會非常合適且一定會提升性能。但假若需要從test表查詢放到緩存里的數據量非常大,那就會出現一個問題:由于數據量過大可能會導致內存不足,而不單單是提升性能了。假如說把表中所有數據都放入緩存,那么緩存的可能會占據大部分jvm的內存或者索性直接產生一個OOM錯誤。
解決方案
最佳的方案是如果我們可以創造一種可以按需擴展和收縮的動態緩存,當我們的數據量需要而內存充裕的時候可以適當增加,但內存不足是可以按不同方案對其進行回收。
目的
這里引出的一個問題,就是為什么要在Java中使用不同類型的reference?我們的應用在運行過程中會產生很多對象,這些對象駐留在內存中,它們大小不同,重要性不同,使用頻率不同,生命周期不同,比如有些對象只要應用啟動就一直存活直到應用停止,而有些對象生命周期與創建它的線程相同,還有些對象只作臨時變量短時間就消亡,再比如某些緩存數據,內存充裕的時候可以存活,內存不足的時候可能需要被首先犧牲被回收,所以很容易想象對于不同的對象,我們希望對他們的創建銷毀采取不同的策略,可是不幸的是java不像C一樣可以由開發者決定對象的析構銷毀,而是將管理內存的活統一交給了jvm進行gc,但jvm顯然不知道這些對象的區別。
于是設計者們在java 1.2加入了reference,使jvm可以對不同的reference對象采取不同的回收策略以達到提高應用性能的目的。
java.lang.ref 包
實際上java.lang.ref包中就有以下幾種不同的reference類型,分別是:
StrongReference
SoftReference
WeakReference
PhantomReference
FinalReference
StrongReference
我們發現在類圖中我們并沒有發現 StrongReference 類型,原因是我們平時寫的代碼基本上都是 StrongReference 。我們最常的創建對象方式就是 new 一個對象,然后將其賦值給一個聲明為這個對象的類型及其父類的引用。如果對象有一個 StrongReference ,那么這個對象將不會被gc回收。
舉例
HelloWorld hello = new HelloWorld();
這里 hello 就是一個 HelloWorld 對象的 StrongReference。
SoftReference
如果一個對象沒有 StrongReference 但存在一個 SoftReference ,那么 gc 將會在虛擬機需要釋放一些內存的時候回收這個對象。可以通過對對象的 SoftReference 調用 get() 方法獲取該對象。如果這個對象沒有被 gc 回收,則返回此對象,否則返回 null 。
WeakReference
如果一個對象沒有 StrongReference 但有存在一個 WeakReference ,那么 gc 將會在下一次運行時對其進行回收,哪怕虛擬機的內存還足夠多。
PhantomReference 與 FinalReference
如果某個對象沒有以上這些類型的引用,那么它可能有一個 PhantomReference 。PhantomReference 不能用于直接訪問對象。調用 get() 方法都會返回 null 。
FinalReference 與虛擬機密切相關,這里先挖個坑,下次再具體解析。
對象可達性判斷
當前主流java虛擬機都是采用 GC Roots Tracing 算法,比如 Sun 的 Hotspot 虛擬機便是采用該算法。java虛擬機進行gc時,判斷一個對象的被引用情況決定是否回收,都是從根節點引用(Root set of Reference)開始標識可達路徑的。對于某個對象可能會存在其多個引用,且這多個引用的類型不同。
如下圖所示:
Root Tracing 算法根據以下兩個原則標記對象的可達性:
單一路徑中,以最弱的引用為準
多路徑中,以最強的引用為準
如上圖所示,對對象4存在3條引用路徑:(1)(4),(2)(5),(3)(6)。那么從根對象到對象4的最強引用時(2)(5),因為(2)和(5)都是強引用。如果對象4僅存在一條(1)(4)引用,那么對它的引用就是最弱的引用為準,也就是 SoftReference ,對象4就是 softly-reachable 對象。
不同類型 reference java 代碼舉例
package com.example.reference;import java.lang.ref.PhantomReference;import java.lang.ref.ReferenceQueue;import java.lang.ref.SoftReference;import java.lang.ref.WeakReference;public class ReferenceExample { private String status ='Hi I am active'; public String getStatus() { return status; } public void setStatus(String status) { this.status = status; } @Override public String toString() { return 'ReferenceExample [status=' + status + ']'; } public void strongReference() { ReferenceExample ex = new ReferenceExample(); System.out.println(ex); } public void softReference() { SoftReference<ReferenceExample> ex = new SoftReference<ReferenceExample>(getRefrence()); System.out.println('Soft refrence :: ' + ex.get()); } public void weakReference() { int counter=0; WeakReference<ReferenceExample> ex = new WeakReference<ReferenceExample>(getRefrence()); while(ex.get()!=null) { counter++; System.gc(); System.out.println('Weak reference deleted after:: ' + counter + ex.get()); } } public void phantomReference() throws InterruptedException { final ReferenceQueue queue = new ReferenceQueue(); PhantomReference<ReferenceExample> ex = new PhantomReference<ReferenceExample>(getRefrence(),queue); System.gc(); queue.remove(); System.out.println('Phantom reference deleted after'); } private ReferenceExample getRefrence() { return new ReferenceExample(); } public static void main(String[] args) { ReferenceExample ex = new ReferenceExample(); ex.strongReference(); ex.softReference(); ex.weakReference(); try { ex.phantomReference(); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } }}Output :ReferenceExample [status=Hi I am active]Soft refrence :: ReferenceExample [status=Hi I am active]Weak reference deleted after:: 1nullPhantom reference deleted after
總結
通過對以上各類型的 reference 介紹可以發現其實 reference 主要是用來與虛擬機 gc 進行交互,使得虛擬機根據對象的不同引用類型,對其采用不同的內存回收策略。
strong 引用的對象正常情況下不會被回收,soft 引用的對象會在出現 OOM 錯誤之前被回收,而 weak 引用的對象在下一次 gc 的時候就會被回收,對 reference 的基本理解就差不多了。
至于 PhantomReference 與 FinalReference 下次再講。
以上這篇java中的Reference類型用法說明就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持好吧啦網。
相關文章: