詳解Java的引用類型及使用場景
每種編程語言都有自己操作內(nèi)存中元素的方式,例如在 C 和 C++ 里是通過指針,而在 Java 中則是通過“引用”。在 JDK.1.2 之后,Java 對(duì)引用的概念進(jìn)行了擴(kuò)充,將引用分為了:強(qiáng)引用(Strong Reference)、軟引用(Soft Reference)、弱引用(Weak Reference)、虛引用(Phantom Reference)4 種,這 4 種引用的強(qiáng)度依次減弱,今天這篇文章就簡單介紹一下這四種類型,并簡單說一下他們的使用場景。
1. 強(qiáng)引用(Strong Reference)
強(qiáng)引用類型,是我們最常講的一個(gè)類型,我們先看一個(gè)例子:
package cn.bridgeli.demo.reference;
/**
* @author BridgeLi
* @date 2021/2/26 10:02
*/
public class User {
@Override
protected void finalize() throws Throwable {
super.finalize();
System.out.println("finalize");
}
}
package cn.bridgeli.demo.reference;
import org.junit.Test;
/**
* @author BridgeLi
* @date 2021/2/26 10:03
*/
public class StrongReferenceTest {
@Test
public void testStrongReference() {
User user = new User();
user = null;
System.gc();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
我們都知道當(dāng)一個(gè)實(shí)例對(duì)象具有強(qiáng)引用時(shí),垃圾回收器不會(huì)回收該對(duì)象,當(dāng)內(nèi)存不足時(shí),寧愿 OOM,也就是拋出 OutOfMemeryError 異常也不會(huì)回收強(qiáng)引用的對(duì)象,因?yàn)?JVM 認(rèn)為強(qiáng)引用的對(duì)象是用戶正在使用的對(duì)象,它無法分辨出到底該回收哪個(gè),強(qiáng)行回收有可能導(dǎo)致系統(tǒng)嚴(yán)重錯(cuò)誤。但是當(dāng)對(duì)象被賦值為 null 之后,會(huì)被回收,并且會(huì)執(zhí)行對(duì)象的 finalize 函數(shù),此時(shí)我們可以通過該函數(shù)拯救自己,但是有兩點(diǎn)需要注意一個(gè)是只能拯救一次,當(dāng)再次被垃圾回收的時(shí)候就不能拯救了,另一個(gè)就是有事沒事千萬不要重寫次函數(shù),本例只是為了說明問題重寫了此函數(shù),如果在工作中誤重寫了此函數(shù),可能會(huì)導(dǎo)致垃圾不能回收,最終 OOM,另外有熟悉 GC 的同學(xué)沒?猜一下我為什么要 sleep 一下?
2. 軟引用(Soft Reference)
在我剛學(xué) Java 的時(shí)候,并不知道怎么使用軟引用,那時(shí)候只知道強(qiáng)引用,其實(shí)是通過 java.lang.ref.SoftReference 類來使用軟引用的,為了說明軟引用,我們先看一個(gè)例子:
package cn.bridgeli.demo.reference;
import org.junit.Test;
import java.lang.ref.SoftReference;
/**
* @author BridgeLi
* @date 2021/2/26 10:21
*/
public class SoftReferenceTest {
@Test
public void testSoftReference() {
SoftReference<byte[]> softReference = new SoftReference<>(new byte[1024 * 1024 * 10]);
System.out.println(softReference.get());
System.gc();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(softReference.get());
byte[] bytes = new byte[1024 * 1024 * 12];
System.out.println(softReference.get());
}
}
除了通過 get 方法獲取我們的軟引用對(duì)象之外,運(yùn)行結(jié)果和強(qiáng)引用類型并沒有什么區(qū)別是吧?結(jié)果和我們想的一樣,但是別著急,加一個(gè)啟動(dòng)參數(shù)再試試:
-Xms20m -Xmx20m
我們都知道,這兩個(gè)參數(shù)是控制 JVM 啟動(dòng)的時(shí)候堆的最大值和最小值的,這里面我們?cè)O(shè)置的最大值和最小值都是 20M,按照強(qiáng)引用的邏輯,我們一共申請(qǐng)了 22M 的空間,應(yīng)該 OOM 才對(duì),事實(shí)證明并沒有,通過打印語句證明,我們的軟引用被回收了,所以軟引用的特點(diǎn)是:在內(nèi)存足夠的時(shí)候,軟引用對(duì)象不會(huì)被垃圾回收器回收,只有在內(nèi)存不足時(shí),垃圾回收器則會(huì)回收軟引用對(duì)象,當(dāng)然回收了軟引用對(duì)象之后仍然沒有足夠的內(nèi)存,這時(shí)同樣會(huì)拋出內(nèi)存溢出異常。
看了軟引用的特點(diǎn),我們很容易想到軟引用的使用場景:緩存。記得剛工作的時(shí)候,有個(gè)同事給我說,他做 Android,有一個(gè)加載圖片的應(yīng)用,特麻煩,會(huì) OOM,其實(shí)使用軟引用應(yīng)該很輕松的能解決這個(gè)問題。
3. 弱引用(Weak Reference)
弱引用是通過 java.lang.ref.WeakReference 類來實(shí)現(xiàn)的,同樣我們也先看一個(gè)例子:
package cn.bridgeli.demo.reference;
import org.junit.Test;
import java.lang.ref.WeakReference;
/**
* @author BridgeLi
* @date 2021/2/26 10:30
*/
public class WeakReferenceTest {
@Test
public void testWeakReference() {
WeakReference<User> weakReference = new WeakReference<>(new User());
System.out.println(weakReference.get());
System.gc();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(weakReference.get());
}
}
通過例子我們可以看到,弱引用是一種比軟引用更弱的引用類型:在系統(tǒng) GC 時(shí),只要發(fā)現(xiàn)弱引用,不管系統(tǒng)堆空間是否足夠,都會(huì)將對(duì)象進(jìn)行回收??吹竭@里可能會(huì)有同學(xué)有疑問,GC 什么時(shí)候啟動(dòng),除了我們顯示調(diào)用外,我們并不能控制(其實(shí)就算我們顯示調(diào)用,GC 也可能不會(huì)立即執(zhí)行),而且 GC 之后,弱引用立即被回收,引用不到了,那么這個(gè)類型有什么用呢?其實(shí)這個(gè)類型還真有大用,我們鼎鼎大名的 ThreadLocal 類就是借助于這個(gè)類實(shí)現(xiàn)的,所以當(dāng)你使用 ThreadLocal 的時(shí)候,就已經(jīng)在使用弱類型了,我之前曾經(jīng)寫過關(guān)于 ThreadLocal 的文章,但是當(dāng)時(shí)理解不是很準(zhǔn)確,不過說明的例子是沒有問題的,所以還有一定的參考價(jià)值,后面看看啥時(shí)候有機(jī)會(huì)重寫一篇關(guān)于 ThreadLocal 的文章,詳細(xì)說說這個(gè)類。
另外除了 ThreadLocal 類外還有一個(gè)類值得說一下,那就是 java.util.WeakHashMap 類,見名知意,我們就可以猜到這個(gè)類的特點(diǎn)。同樣通過一個(gè)例子說明一下:
package cn.bridgeli.demo.reference;
import org.junit.Test;
import java.util.Map;
import java.util.WeakHashMap;
/**
* @author BridgeLi
* @date 2021/2/26 10:38
*/
public class WeakHashMapTest {
@Test
public void testWeakHashMap() {
Map map = new WeakHashMap<String, Object>();
for (int i = 0; i < 10000; i++) {
map.put("key" + i, new byte[i]);
}
// Map map = new HashMap<String, Object>();
// for (int i = 0; i < 10000; i++) {
// map.put("key" + i, new byte[i]);
// }
}
}
記得啟動(dòng)的時(shí)候設(shè)置一下,設(shè)置一下啟動(dòng)的時(shí)候堆的大小,不要設(shè)置太大,可以看出區(qū)別。
4. 虛引用(Phantom Reference)
通過前面的例子,我們可以看到引用強(qiáng)度是越來越弱的,所以虛引用是最弱的一種引用類型,到底有多弱呢,我們同樣通過一個(gè)例子來看,需要說明的是,虛引用是通過 java.lang.ref.PhantomReference 類實(shí)現(xiàn)的。
package cn.bridgeli.demo.reference;
import org.junit.Test;
import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.util.ArrayList;
import java.util.List;
/**
* @author BridgeLi
* @date 2021/2/26 11:05
*/
public class PhantomReferenceTest {
ReferenceQueue referenceQueue = new ReferenceQueue();
List<Object> list = new ArrayList<>();
@Test
public void testPhantomReference() {
PhantomReference<Object> phantomReference = new PhantomReference<>(new Object(), referenceQueue);
System.out.println(phantomReference.get());
new Thread(() -> {
while (true) {
Reference reference = referenceQueue.poll();
if (null != reference) {
System.out.println("============ " + reference.hashCode() + " ============");
}
}
}).start();
new Thread(() -> {
while (true) {
list.add(new byte[1024 * 1024 * 10]);
}
}).start();
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
我們看到了是什么?雖然軟引用和弱引用也很弱,但是我們還是可以通過 get 方法獲取到我們的引用對(duì)象,但是虛引用卻不行,點(diǎn)進(jìn)去看一下源碼,我們可以看到虛引用的 get 方法,直接返回 null,也就是我們直接拿不到虛引用對(duì)象,那么這個(gè)類型又有什么使用場景呢?其實(shí)這個(gè)類型就不是給我們普通程序員使用的,在 io、堆外內(nèi)存中有使用,所以對(duì)于我們普通程序員來說,了解到存在這個(gè)類型,另外通過上面的例子,我們還可以看到:當(dāng)垃圾回收器準(zhǔn)備回收一個(gè)對(duì)象時(shí),如果發(fā)現(xiàn)它還有虛引用,就會(huì)在垃圾回收后,銷毀這個(gè)對(duì)象,將這個(gè)虛引用加入引用隊(duì)列。程序可以通過判斷引用隊(duì)列中是否已經(jīng)加入了虛引用,來了解被引用的對(duì)象是否將要被垃圾回收。那么我們就可以在程序中發(fā)現(xiàn)某個(gè)虛引用已經(jīng)被加入到引用隊(duì)列,那么就可以在所引用的對(duì)象的內(nèi)存被回收之前采取一些必要的行動(dòng)。
以上就是詳解Java的引用類型及使用場景的詳細(xì)內(nèi)容,更多關(guān)于Java 引用類型及使用場景的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java生成jar包并且單進(jìn)程運(yùn)行的實(shí)例
下面小編就為大家分享一篇java生成jar包并且單進(jìn)程運(yùn)行的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2017-12-12
java實(shí)現(xiàn)解析二進(jìn)制文件的方法(字符串、圖片)
本篇文章主要介紹了java實(shí)現(xiàn)解析二進(jìn)制文件的方法(字符串、圖片),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-02-02
Spring Boot項(xiàng)目集成UidGenerato的方法步驟
這篇文章主要介紹了Spring Boot項(xiàng)目集成UidGenerato的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12
使用SpringBoot與Thrift實(shí)現(xiàn)RPC通信的方式詳解
在微服務(wù)架構(gòu)的世界里,服務(wù)間的通信機(jī)制選擇成為了關(guān)鍵決策之一,RPC因其簡潔、高效的特點(diǎn)備受青睞,本文將詳細(xì)探討如何利用Spring?Boot和Thrift框架構(gòu)建RPC通信,讓讀者理解其內(nèi)在原理及實(shí)現(xiàn)方式,需要的朋友可以參考下2023-10-10
AsyncHttpClient的TimeoutTimerTask連接池異步超時(shí)
這篇文章主要為大家介紹了AsyncHttpClient的TimeoutTimerTask連接池異步超時(shí)源碼流程解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12
使用spring-task定時(shí)任務(wù)動(dòng)態(tài)配置修改執(zhí)行時(shí)間
這篇文章主要介紹了使用spring-task定時(shí)任務(wù)動(dòng)態(tài)配置修改執(zhí)行時(shí)間,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
Springboot發(fā)送郵件功能的實(shí)現(xiàn)詳解
電子郵件是—種用電子手段提供信息交換的通信方式,是互聯(lián)網(wǎng)應(yīng)用最廣的服務(wù)。本文詳細(xì)為大家介紹了SpringBoot實(shí)現(xiàn)發(fā)送電子郵件功能的示例代碼,需要的可以參考一下2022-09-09

