Java基礎之引用相關知識總結
一、引用的定義
在JDK 1.2以前,Java中的引用定義很傳統:如果reference類型的數據存儲的數值代表的是另外一塊內存的起始地址,就稱這塊內存代表著一個引用。
二、問題
當描述這樣的一類對象:當內存空間還足夠時,則能保留在內存之中,如果內存空間在進行垃圾收集後還是非常緊張,則可以拋棄這些對象。此時的引用就顯得過於狹隘。因此在JDK 1.2之後,Java堆引用的概念進行瞭擴充。
三、引用的分類
回收時機 | |||
強引用 | Strong Reference | 類似Object obj = new Object() | 隻要強引用還存在,垃圾收集器永遠不會回收掉被引用的對象 |
軟引用 | Soft Reference | 描述一些還有用但並非必須的對象 | 在系統將要發生內存溢出的異常之前,將會把這些對象列進回收范圍之中進行第二次回收。如果這次回收還沒有足夠的內存,才會拋出內存溢出異常 |
弱引用 | Weak Reference | 用來描述非必須對象,強度比軟引用更弱。被弱引用關聯的對象隻能生存到下一次垃圾收集發生之前 | 當垃圾收集器工作時,無論當前內存是否足夠,都會回收掉紙杯弱引用關聯的對象 |
虛引用 | Phantom Reference | 又稱為幽靈引用或幻影引用,最弱的一種引用關系。一個對象是否有虛引用的存在,完全不會對其生存時間構成影響,也無法通過虛引用來取得一個對象實例。 | 為一個對象設置虛引用關聯的衛衣目的就是能在這個對象被收集器回收時收到一個系統通知 |
四、應用場景
1.強引用
最常用的,應用場景最多,everywhere
2.軟引用
隻有將要發生OOM的時候,才會主動回收,應用在一些內存限制比較大的應用內,避免發生OOM。
3.弱引用
不管是young gc 還是 full gc 都會回收弱引用的對象。當你想引用一個對象,但是這個對象有自己的生命周期,你不想介入這個對象的生命周期,這時候你就是用弱引用。
4.虛引用
在回收時可以檢測到,虛引用主要用來跟蹤對象被垃圾回收器回收的活動。
五、源碼
package org.pdool.ref; import java.lang.ref.Reference; import java.lang.ref.ReferenceQueue; import java.lang.ref.WeakReference; import java.util.LinkedList; public class ReferenceTest { private static ReferenceQueue<VeryBig> rq = new ReferenceQueue<VeryBig>(); public static void checkQueue() { Reference<? extends VeryBig> ref = null; while ((ref = rq.poll()) != null) { // 隻留有引用,沒有對象 System.out.println("In queue: " + ((VeryBigWeakReference) (ref)).id); } } public static void main(String args[]) { int size = 3; LinkedList<WeakReference<VeryBig>> weakList = new LinkedList<WeakReference<VeryBig>>(); for (int i = 0; i < size; i++) { weakList.add(new VeryBigWeakReference(new VeryBig("Weak " + i), rq)); System.out.println("Just created weak: " + weakList.getLast()); } System.gc(); try { // 暫停6s,讓上面的垃圾回收線程運行完成 Thread.currentThread().sleep(6000); } catch (InterruptedException e) { e.printStackTrace(); } checkQueue(); } } class VeryBig { public String id; // 占用空間,讓線程進行回收 byte[] b = new byte[2 * 1024]; public VeryBig(String id) { this.id = id; } protected void finalize() { System.out.println("Finalizing VeryBig " + id); } } class VeryBigWeakReference extends WeakReference<VeryBig> { public String id; public VeryBigWeakReference(VeryBig big, ReferenceQueue<VeryBig> rq) { super(big, rq); this.id = big.id; } protected void finalize() { System.out.println("Finalizing VeryBigWeakReference " + id); } }
六、總結
雖然Java提供瞭這麼多引用,但是在平常的業務開發中基本上隻要強引用就可以瞭,整體來說上面除瞭強引用其他三個引用都是和內存相關的,隻要在開發中不要內存泄漏,基本上內存不會出現問題
到此這篇關於Java基礎之引用相關知識總結的文章就介紹到這瞭,更多相關Java引用內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 淺談JVM垃圾回收之哪些對象可以被回收
- Java中四種引用詳解
- Android中深入學習對象的四種引用類型
- WeakHashMap 和 HashMap 區別及使用場景
- java編程Reference核心原理示例源碼分析