java 引用類型: 強(qiáng),軟尤误,弱侠畔,虛,引用對(duì)象的能力依次減弱损晤。
強(qiáng)引用 (FinalReference)
String a = new String ("aaa");
這里 a 即是強(qiáng)引用软棺。我們都知道java對(duì)象是被分配在jvm堆上的,當(dāng)程序中不再有可達(dá)性的引用指向java對(duì)象時(shí)尤勋,該對(duì)象即可被gc回收喘落。而被強(qiáng)引用指向的對(duì)象永遠(yuǎn)不會(huì)被gc。在虛擬機(jī)的實(shí)現(xiàn)過(guò)程中最冰,實(shí)際采用了 FinalReference 類對(duì)其進(jìn)行引用瘦棋。而 Finalizer,除了作為一個(gè)實(shí)現(xiàn)類外暖哨,更是在虛擬機(jī)中實(shí)現(xiàn)一個(gè) FinalizerThread赌朋,以使虛擬機(jī)能夠在所有的強(qiáng)引用被解除后實(shí)現(xiàn)內(nèi)存清理。
static {
ThreadGroup tg = Thread.currentThread().getThreadGroup();
for (ThreadGroup tgn = tg;
tgn != null;
tg = tgn, tgn = tg.getParent());
Thread finalizer = new FinalizerThread(tg);
finalizer.setPriority(Thread.MAX_PRIORITY - 2);
finalizer.setDaemon(true);
finalizer.start();
}
在 GC 的過(guò)程中篇裁,當(dāng)一個(gè)強(qiáng)引用被釋放沛慢,由系統(tǒng)垃圾收集器標(biāo)記后的對(duì)象,會(huì)被加入 Finalizer 對(duì)象中的 ReferenceQueue 中去达布,并調(diào)用 Finalizer.runFinalizer() 來(lái)執(zhí)行對(duì)象的 finalize 方法团甲。
private void runFinalizer(JavaLangAccess jla) {
synchronized (this) {
if (hasBeenFinalized()) return;
remove();
}
try {
Object finalizee = this.get();
if (finalizee != null && !(finalizee instanceof java.lang.Enum)) {
jla.invokeFinalize(finalizee);
/* 注意,這里需要清空棧中包含該變量的的 slot, 從而來(lái)減少因?yàn)橐粋€(gè)保守的 GC 實(shí)現(xiàn)所造成的變量未被回收的假象 */
finalizee = null;
}
} catch (Throwable x) { }
super.clear();
}
注意往枣,標(biāo)記處所調(diào)用的 invokeFinalizeMethod 為 native 方法伐庭,由于 finalize 方法在 Object 類中被聲明為 protected粉渠,這里必須采用 native 方法才能調(diào)用。隨后通過(guò)將本地強(qiáng)引用設(shè)置為空圾另,以便使垃圾回收器清理內(nèi)存霸株。
強(qiáng)引用有以下特征:
可直接訪問(wèn)對(duì)象
被強(qiáng)引用指向的對(duì)象即便在gc時(shí)也不會(huì)被回收,系統(tǒng)寧可發(fā)生OOM
強(qiáng)引用可能導(dǎo)致內(nèi)存泄漏(這里補(bǔ)充一下內(nèi)存泄漏與內(nèi)存溢出的區(qū)別: 內(nèi)存泄漏是指本該被回收的對(duì)象沒(méi)有被回收集乔,內(nèi)存溢出是指內(nèi)存容量已達(dá)上限去件,再往里加就溢出了,就像是水桶滿了再加水就溢到外面了)
軟引用(SoftReference)
SoftReference<Integer> softReference = new SoftReference<>(1);
softReference.get(); // 1
軟引用指向?qū)ο蟮哪芰Υ斡趶?qiáng)引用扰路,當(dāng)內(nèi)存容量充足時(shí)并不會(huì)對(duì)軟引用所指向的對(duì)象有何影響尤溜,但是當(dāng)內(nèi)存容量不足時(shí),將會(huì)回收其指向的對(duì)象汗唱。首先會(huì)清空它的softReference宫莱,也就是softReference.get()返回null,然后再調(diào)用對(duì)象的finalize()方法哩罪,并在下一輪的gc中對(duì)其做真正的回收授霸。軟引用有以下特征:
- 通過(guò)get()取得對(duì)象強(qiáng)引用來(lái)訪問(wèn)對(duì)象
- 內(nèi)存吃緊情況下將會(huì)被回收
弱引用(WeakReference)
WeakReference<Integer> weakReference = new WeakReference<>(1);
weakReference.get(); // 1
弱引用指向?qū)ο蟮哪芰τ秩跤谲浺谩F涮匦钥绍浺没鞠嗨萍什澹煌氖?strong>不管內(nèi)存容量是否充足碘耳,只要發(fā)生了gc,都會(huì)對(duì)其進(jìn)行回收框弛。即get() 返回的都是null辛辨。 弱引用有以下特征:
- 通過(guò)get()取得對(duì)象強(qiáng)引用來(lái)訪問(wèn)對(duì)象
- 不管內(nèi)存是否吃緊,發(fā)生gc都會(huì)被回收
虛引用(PhantomReference)
ReferenceQueue queue = new ReferenceQueue();
PhantomReference phantomReference = new PhantomReference<>("aaa" ,queue);
虛引用是“最弱的” 引用類型瑟枫。它的構(gòu)造函數(shù)需要一個(gè)ReferenceQueue參數(shù)而且它不能通過(guò)get來(lái)獲取對(duì)象斗搞,因?yàn)樗膅et() 方法返回的永遠(yuǎn)是null。
public T get() { return null; }
那虛引用的作用是什么呢力奋? 其實(shí)虛引用主要是被用來(lái) 跟蹤對(duì)象被垃圾回收的狀態(tài)榜旦,通過(guò)查看引用隊(duì)列中是否包含對(duì)象的虛引用來(lái)判斷它是否即將被回收。目標(biāo)對(duì)象被gc回收前景殷,虛引用會(huì)被放入一個(gè)ReferenceQueue對(duì)象中,從而達(dá)到跟蹤對(duì)象被gc的作用澡屡。虛引用有以下特征:
- 虛引用無(wú)法通過(guò)get()方法獲取對(duì)象
- 虛引用所指向的對(duì)象在被gc前猿挚,會(huì)將虛引用放一個(gè)ReferenceQueue中從面跟蹤對(duì)象gc
值得注意的是,SoftReference, WeakReference 以及 PhantomReference 的構(gòu)造函數(shù)都可以接收一個(gè) ReferenceQueue 對(duì)象驶鹉。但是SoftReference, WeakReference在他們所指向的對(duì)象準(zhǔn)備被gc回收時(shí)绩蜻,調(diào)用對(duì)象finalize()方法之前,它們自身都會(huì)被放入這個(gè)queue中室埋。而PhantomReference只有當(dāng)gc對(duì)其所指向的對(duì)象真正進(jìn)行回收時(shí)办绝,才會(huì)放入這個(gè)queue中伊约。
總結(jié):
強(qiáng)引用 (FinalReference), 這是最常用的引用類型 . JVM 系統(tǒng)采用 Finalizer 來(lái)管理每個(gè)強(qiáng)引用對(duì)象 , 并將其被標(biāo)記要清理時(shí)加入 ReferenceQueue, 并逐一調(diào)用該對(duì)象的 finalize() 方法 .
軟引用 (SoftReference), 引用類型表現(xiàn)為當(dāng)內(nèi)存接近滿負(fù)荷 , 或?qū)ο笥?SoftReference.get() 方法的調(diào)用沒(méi)有發(fā)生一段時(shí)間后 , 垃圾回收器將會(huì)清理該對(duì)象 . 在運(yùn)行對(duì)象的 finalize 方法前 , 會(huì)將軟引用對(duì)象加入 ReferenceQueue 中去 .
弱引用 (WeakReference), 引用類型表現(xiàn)為當(dāng)系統(tǒng)垃圾回收器開(kāi)始回收時(shí) , 則立即會(huì)回收該對(duì)象的引用 . 與軟引用一樣 , 弱引用也會(huì)在運(yùn)行對(duì)象的 finalize 方法之前將弱引用對(duì)象加入 ReferenceQueue.
虛引用 (PhantomReference), 這是一個(gè)最虛幻的引用類型 . 無(wú)論是從哪里都無(wú)法再次返回被虛引用所引用的對(duì)象 . 虛引用在系統(tǒng)垃圾回收器開(kāi)始回收對(duì)象時(shí) , 將直接調(diào)用 finalize() 方法 , 但不會(huì)立即將其加入回收隊(duì)列 . 只有在真正對(duì)象被 GC 清除時(shí) , 才會(huì)將其加入 Reference 隊(duì)列中去 .