JVM中如何理解强引用、软引用、弱引用、虚引用?

整体架构

强引用

强引用是默认支持,当内存不足的时候,JVM开始垃圾回收,对于强引用的对象,就算是出现了OOM也不会回收对象。

强引用是最常见的普通对象引用,只要还有强引用指向对象,对象就存活,垃圾回收器不会处理存活对象。一般把一个对象赋给一个引用变量,这个引用变量就是强引用。当一个对象被强引用变量所引用,它就处于可达状态,是不会被垃圾回收的,即使之后都不会再用到了,也不会回收。因此强引用是造成Java内存泄漏的主要原因之一。

对于一个普通对象,如果没有其他引用关系,只要超过了引用的作用域或者显式地将相应的强引用赋值为null,一般认为就是可以被垃圾回收了。(具体的回收时机看垃圾回收策略)

下例中,b就是强引用。

1     public static void main(String[] args) { 
2         Object a = new Object(); 
3         Object b = a; 
4         a = null; 
5         System.out.println(b);//java.lang.Object@4554617c
6     }

软引用

软引用是一种相对强引用弱化了一些的引用,用java.lang.ref.SoftReference实现,可以让对象豁免一些垃圾收集。当系统内存充足的时候,不会被回收;当系统内存不足的时候,会被回收。

软引用一般用于对内存敏感的程序中,比如高速缓存。

1 import java.lang.ref.SoftReference; 
2 
3 public class SoftReferenceDemo { 
4     public static void main(String[] args) { 
5         Object a = new Object(); 
6         SoftReference<Object> softReference = new SoftReference<>(a);//软引用
7         //a和软引用指向同一个对象
8         System.out.println(a);//java.lang.Object@4554617c
9         System.out.println(softReference.get());//java.lang.Object@4554617c 10 
11         //内存够用,软引用不会被回收
12         a = null; 13         System.gc();//内存够用不会自动gc,手动唤醒gc
14         System.out.println(a);//null
15         System.out.println(softReference.get());//java.lang.Object@4554617c 16 
17         //内存不够用时
18         try{ 19             //配置Xms和Xmx为5MB
20             byte[] bytes = new byte[1024*1024*30];//设置30MB超内存
21         }catch (Throwable e){ 22 e.printStackTrace(); 23         }finally { 24             System.out.println(a);//null
25             System.out.println(softReference.get());//null
26 } 
27 }
28 }

使用场景

一个应用需要读取大量的本地图片,如果每次读取都从硬盘读取会严重影响性能,如果一次性全部加载到内存,内存可能会溢出。

可以使用软引用解决这个问题,使用一个HashMap来保存图片路径和图片对象管理的软引用之间的映射关系,内存不足时,JVM会自动回收缓存图片对象的占用空间,有效地避免了OOM(Out Of Memory)问题。

Map<String, SoftReference<Bitmap>> imageCache = new HashMap<String, SoftReference<Bitmap>>

弱引用

弱引用需要用java.lang.ref.WeakReference实现,它比软引用的生存期更短,对于弱引用的对象来说,只要垃圾回收机制一运行,不管JVM的内存空间是否够,都会回收该对象的占用内存。

1 import java.lang.ref.WeakReference; 
2 
3 public class SoftReferenceDemo { 
4     public static void main(String[] args) { 
5         Object a = new Object(); 
6         WeakReference<Object> softReference = new WeakReference<>(a);//软引用
7         //a和弱引用指向同一个对象
8         System.out.println(a);//java.lang.Object@4554617c
9         System.out.println(softReference.get());//java.lang.Object@4554617c 10 
11         //内存够用,弱引用也会被回收
12         a = null; 13         System.gc();//内存够用不会自动gc,手动唤醒gc
14         System.out.println(a);//null
15         System.out.println(softReference.get());//null
16 } 
17 }

关于WeakHashMap

1     public static void weakHashMapTest() { 
2         Integer key = new Integer(1);
3         String value = "李四";
4         Map<Integer,String> weakHashMap = new WeakHashMap(); 
5         weakHashMap.put(key, value);
6         System.out.println(weakHashMap);//{1=李四}
7         key = null;
8         System.gc();
9         System.out.println(weakHashMap);//{}10 } 
11 
12     public static void hashMapTest() { 
13         HashMap<Integer,String> map = new HashMap<>(); 
14         Integer key = 1; 
15         String value = "张三"; 
16 map.put(key,value); 
17         System.out.println(map);//{1=张三}
18         key = null; 
19 System.gc(); 
20         System.out.println(map);//{1=张三}
21     }

在HashMap中,键被置为null,唤醒gc后,不会垃圾回收键为null的键值对。但是在WeakHashMap中,键被置为null,唤醒gc后,键为null的键值对会被回收。

虚引用

虚引用要通过java.lang.ref.PhantomReference类来实现,虚引用不会决定对象的生命周期,如果一个对象只有虚引用,就相当于没有引用,在任何时候都可能会被垃圾回收器回收。它不能单独使用也不能访问对象,虚引用必须和引用队列联合使用。

虚引用的主要作用是跟踪对象被垃圾回收的状态,仅仅是提供一种确保对象被finalize以后,做某些事情的机制。

PhantomReference的get方法总是返回null,因此无法访问对应的引用对象,设置虚引用关联唯一的目的是在对象被收集器回收的时候收到一个系统通知,或者后续添加进一步的处理。Java允许使用finalize()方法在垃圾回收器将对象从内存中清理出去之前做一些必要的清理工作。【例如实现一个监控对象的通知机制】

引用队列

WeakReference和ReferenceQueue的联合使用效果:

1     public static void weakReferenceTest() { 
2         Object a = new Object(); 
3         ReferenceQueue<Object> queue = new ReferenceQueue<>();
4         WeakReference<Object> weakReference = new WeakReference<>(a,queue);
5         System.out.println(a);//java.lang.Object@4554617c
6         System.out.println(weakReference.get());//java.lang.Object@4554617c
7         System.out.println(queue.poll());//null
8         System.out.println("-------------------");
9         a = null; 
10 System.gc(); 
11         System.out.println(a);//null
12         System.out.println(weakReference.get());//null 13         //虚引用在回收之前被加入到了引用队列中
14         System.out.println(queue.poll());//java.lang.ref.WeakReference@74a14482
15     }

PhantomReference和ReferenceQueue的联合使用效果:

1     public static void phantomReferenceTest() { 
2         Object a = new Object(); 
3         ReferenceQueue<Object> queue = new ReferenceQueue<>();
4         PhantomReference<Object> phantomReference = new PhantomReference<>(a,queue);
5         System.out.println(a);//java.lang.Object@4554617c
6         System.out.println(phantomReference.get());//null
7         System.out.println(queue.poll());//null
8         System.out.println("-------------------");
9         a = null; 
10 System.gc(); 
11         System.out.println(a);//null
12         System.out.println(phantomReference.get());//null 
13         //引用在回收之前被加入到了引用队列中
14         System.out.println(queue.poll());//java.lang.ref.WeakReference@74a14482
15     }

总结

强引用:不回收。

软引用:内存不够就回收。

弱引用:一定回收。

虚引用:一定回收,get出来就是null,引用形同虚设,主要和引用队列联合使用,在finalize之前会被放到引用队列中。

与根对象没有引用关系的:引用不可达,一定回收。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343