弱引用,软引用及虚引用对GC的影响

标签: 弱引用 引用 引用 | 发表时间:2015-12-17 09:40 | 作者:
出处:http://it.deepinmind.com

在应用程序中使用非强引用会导致一系列的问题,对GC的响应时间及吞吐量都会有所影响。尽管这类引用在某些场景下可以减少 OutOfMemoryError的出现频率,但过度的使用则会严重影响到垃圾回收,从而导致应用程序的性能遭受影响。

应该注意什么?

在使用弱引用时,你最好了解一下它是如何被回收的。一旦垃圾回收器发现有一个对象是弱可达的,也就是说,它只剩下弱引用了,这个对象就会被放到一个相应的ReferenceQueue中,这样它就可以被析构并回收了。随后它可能会被从队列中取出,并进行相关的清理活动。这类清理任务的一个典型例子便是清除缓存中的无效KEY。

这里还有一个小技巧就是在这个对象最终被回收前,你还能再新建一个强引用来指向它,因此垃圾回收器在回收前还必须得进行二次确认。

其实弱引用要比你想象的常见得多。许多缓存方案都是通过弱引用来解决的,所以即便你没有在代码中创建过弱引用对象,但很可能你的应用程序中就大量使用到了它们。

而说到软引用,你需要知道的是它的回收速度要比弱引用慢一些。它具体会何时回收是不确定的,这取决于具体的JVM实现。通常来说在多次回收内存后空间仍然不足的话会触发软引用的回收。也就是说,你可能会面临比预期的更频繁且时间更长的GC暂停。

而对于虚引用,你可能就得手动进行内存管理并将它们标记为允许回收才行了。虚引用非常危险,如果你只是简单地看一眼文档的话,你会觉得使用它是非常安全的:

为了确保可回收对象保持原状,虚引用指向的对象必须无法被获取到:虚引用的get方法应当始终返回null。

但很奇怪的是,许多开发人员都忽略了该文档中接下来的一段说明(加粗部分):

与弱引用和软引用不同,虚引用在加入回收队列后,是无法被垃圾回收器自动清除的。虚引用可达的对象会一直维持原状,直到这类引用都被清除或者它们自身也不可达。

没错,我们必须手动地 clear())虚引用,否则便会让JVM陷入内存耗尽的处境。虚引用存在的意义首先就在于这是确定对象是否已经不可达的唯一途径。与弱引用和软引用不同,虚引用对象是决不可能“复活”的。

看几个例子

我们来看下这个 demo程序,它会创建大量的对象,但在新生代GC中就会被回收掉。但是还有个调整持久代的提升比率阈值的小窍门,让我们用-Xmx24m -XX:NewSize=16m -XX:MaxTenuringThreshold=1这个参数来运行下这个程序,来看看它的GC日志:

   2.330: [GC (Allocation Failure)  20933K->8229K(22528K), 0.0033848 secs]
2.335: [GC (Allocation Failure)  20517K->7813K(22528K), 0.0022426 secs]
2.339: [GC (Allocation Failure)  20101K->7429K(22528K), 0.0010920 secs]
2.341: [GC (Allocation Failure)  19717K->9157K(22528K), 0.0056285 secs]
2.348: [GC (Allocation Failure)  21445K->8997K(22528K), 0.0041313 secs]
2.354: [GC (Allocation Failure)  21285K->8581K(22528K), 0.0033737 secs]
2.359: [GC (Allocation Failure)  20869K->8197K(22528K), 0.0023407 secs]
2.362: [GC (Allocation Failure)  20485K->7845K(22528K), 0.0011553 secs]
2.365: [GC (Allocation Failure)  20133K->9501K(22528K), 0.0060705 secs]
2.371: [Full GC (Ergonomics)  9501K->2987K(22528K), 0.0171452 secs]

这种情况下是几乎没有Full GC的。然而,一旦程序开始创建弱引用的话(-Dweak.refs=true),情况将彻底发生变化。这种情况很常见,比如说用作弱哈希表的KEY或者分析对象创建的性能。但不管是什么情况,弱引用的引入都会导致这样的结果:

   2.059: [Full GC (Ergonomics)  20365K->19611K(22528K), 0.0654090 secs]
2.125: [Full GC (Ergonomics)  20365K->19711K(22528K), 0.0707499 secs]
2.196: [Full GC (Ergonomics)  20365K->19798K(22528K), 0.0717052 secs]
2.268: [Full GC (Ergonomics)  20365K->19873K(22528K), 0.0686290 secs]
2.337: [Full GC (Ergonomics)  20365K->19939K(22528K), 0.0702009 secs]
2.407: [Full GC (Ergonomics)  20365K->19995K(22528K), 0.0694095 secs]

正如你所看到的,现在出现了大量的Full GC,并且回收的时间也增加了一个数量级。显然这是一次过早提升(premature promotion),不过却有点棘手。当然,罪魁祸首就是弱引用。在没使用它们之前,应用所创建的对象在提升到老生代前就已经被回收掉了。但是使用了弱使用之后,这些对象会多存活一次GC周期才能被正确地回收掉。一个简单的解决方案就是增加新生代的大小,-Xmx64m -XX:NewSize=32m:

   2.328: [GC (Allocation Failure)  38940K->13596K(61440K), 0.0012818 secs]
2.332: [GC (Allocation Failure)  38172K->14812K(61440K), 0.0060333 secs]
2.341: [GC (Allocation Failure)  39388K->13948K(61440K), 0.0029427 secs]
2.347: [GC (Allocation Failure)  38524K->15228K(61440K), 0.0101199 secs]
2.361: [GC (Allocation Failure)  39804K->14428K(61440K), 0.0040940 secs]
2.368: [GC (Allocation Failure)  39004K->13532K(61440K), 0.0012451 secs]

于是这些对象便又能在新生代GC中被回收掉了。

下一个DEMO程序中我们用到了软引用,而情况就更为严重了。除非应用面临即将抛出OutOfMemoryError异常的风险,否则软可达的对象是不会被回收的。用软引用替换掉弱引用之后,DEMO程序立即便出现了更多的FULL GC事件:

   2.162: [Full GC (Ergonomics)  31561K->12865K(61440K), 0.0181392 secs]
2.184: [GC (Allocation Failure)  37441K->17585K(61440K), 0.0024479 secs]
2.189: [GC (Allocation Failure)  42161K->27033K(61440K), 0.0061485 secs]
2.195: [Full GC (Ergonomics)  27033K->14385K(61440K), 0.0228773 secs]
2.221: [GC (Allocation Failure)  38961K->20633K(61440K), 0.0030729 secs]
2.227: [GC (Allocation Failure)  45209K->31609K(61440K), 0.0069772 secs]
2.234: [Full GC (Ergonomics)  31609K->15905K(61440K), 0.0257689 secs]

第三个DEMO程序中不难看出,虚引用是当之无愧的王者。使用相同的JVM参数运行这个程序,结果和弱引用的非常类似。由于我们在本节前面中所提到的两者在回收阶段的区别,因此事实上这里出现的Full GC暂停要更少一些。

不过,一旦加上停止清除虚引用的标记(-Dno.ref.clearing=true)之后,结果马上就变成了这样:

   4.180: [Full GC (Ergonomics)  57343K->57087K(61440K), 0.0879851 secs]
4.269: [Full GC (Ergonomics)  57089K->57088K(61440K), 0.0973912 secs]
4.366: [Full GC (Ergonomics)  57091K->57089K(61440K), 0.0948099 secs]
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

虚引用的使用一定要慎之又慎,并且一定要及时清理虚可达对象。否则的话,等着你的就会是OutOfMemoryError了。相信我,不这么做的话你很快就会栽在这里:处理引用队列的那个线程抛出了一个异常,而等着你的只是一个挂掉的应用。

我的JVM会不会受到影响?

通常我们会建议使用-XX:+PrintReferenceGC这个JVM选项来分析不同的引用类型对垃圾回收所带来的影响。如果从弱引用的例子开始就加入这个选项的话,你会看到:

   2.173: [Full GC (Ergonomics) 2.234: [SoftReference, 0 refs, 0.0000151 secs]2.234: [WeakReference, 2648 refs, 0.0001714 secs]2.234: [FinalReference, 1 refs, 0.0000037 secs]2.234: [PhantomReference, 0 refs, 0 refs, 0.0000039 secs]2.234: [JNI Weak Reference, 0.0000027 secs][PSYoungGen: 9216K->8676K(10752K)] [ParOldGen: 12115K->12115K(12288K)] 21331K->20792K(23040K), [Metaspace: 3725K->3725K(1056768K)], 0.0766685 secs] [Times: user=0.49 sys=0.01, real=0.08 secs] 
2.250: [Full GC (Ergonomics) 2.307: [SoftReference, 0 refs, 0.0000173 secs]2.307: [WeakReference, 2298 refs, 0.0001535 secs]2.307: [FinalReference, 3 refs, 0.0000043 secs]2.307: [PhantomReference, 0 refs, 0 refs, 0.0000042 secs]2.307: [JNI Weak Reference, 0.0000029 secs][PSYoungGen: 9215K->8747K(10752K)] [ParOldGen: 12115K->12115K(12288K)] 21331K->20863K(23040K), [Metaspace: 3725K->3725K(1056768K)], 0.0734832 secs] [Times: user=0.52 sys=0.01, real=0.07 secs] 
2.323: [Full GC (Ergonomics) 2.383: [SoftReference, 0 refs, 0.0000161 secs]2.383: [WeakReference, 1981 refs, 0.0001292 secs]2.383: [FinalReference, 16 refs, 0.0000049 secs]2.383: [PhantomReference, 0 refs, 0 refs, 0.0000040 secs]2.383: [JNI Weak Reference, 0.0000027 secs][PSYoungGen: 9216K->8809K(10752K)] [ParOldGen: 12115K->12115K(12288K)]

通常来说,只有当你确信自己的应用程序的吞吐量或者响应时延已经受到GC的影响时,才有必要去分析这类信息。只有在这种情况下,你才有必要去检查此类日志。一般情况下,每次GC周期中所回收的引用数都不多,大多数情况下都是0。如果应用程序花费了大量的时间在进行引用的清理或者出现大量引用被回收的情况,那就需要进行进一步的分析了。

解决方案

一旦你确认自己的程序出现了误用、滥用弱引用,软引用或虚引用的情况,这通常就需要调整一下应用的内部实现逻辑了。一般这都得视应用的具体情况而定,很难给出什么通用的建议。不过,应该牢记如下几条原则:

  • 弱引用:如果问题的现象是某个内存池的使用量上升的话,那么通常增加一下池的大小(或话干脆把整个堆调大)就能解决问题。正如上述示例中所见,增加堆的大小或者新生代的大小可以缓解此类问题。
  • 虚引用:一定要确认是否清除了引用。稍不留神便会忽略了某些情况,从而导致清理的线程无法及时地清除引用队列,或者甚至没有清理,从而将负担扔给了垃圾回收器,进而便会导致抛出OutOfMemoryError异常。
  • 软引用:一旦发现软引用是罪魁祸首,那就只能去修改下应用的实现逻辑了。

原创文章转载请注明出处: 弱引用,软引用及虚引用对GC的影响

英文原文链接

相关 [弱引用 引用 引用] 推荐:

java中的强引用、软引用、弱引用、 虚引用

- - 编程语言 - ITeye博客
     今晚接触到一种说法,就是软引用,查了一下,原来还有其他引用. 各种百度和谷歌总算有点头绪,虽然不是很懂,但是总结一下. ⑴强引用(StrongReference). 强引用是使用最普遍的引用. 如果一个对象具有强引用,那垃圾回收器绝不会回收它. 当内存空间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足的问题.

弱引用,软引用及虚引用对GC的影响

- - Java译站
在应用程序中使用非强引用会导致一系列的问题,对GC的响应时间及吞吐量都会有所影响. 尽管这类引用在某些场景下可以减少 OutOfMemoryError的出现频率,但过度的使用则会严重影响到垃圾回收,从而导致应用程序的性能遭受影响. 在使用弱引用时,你最好了解一下它是如何被回收的. 一旦垃圾回收器发现有一个对象是弱可达的,也就是说,它只剩下弱引用了,这个对象就会被放到一个相应的ReferenceQueue中,这样它就可以被析构并回收了.

hibernate引用查询和联合查询

- - ITeye博客
日常开发时常会如下的编码规范:“代码中不允许出现sql语句”,其实这里面的意图其实很明显,sql语句混杂在代码之间会破坏代码的可读性和维护性,此时,有人难免失望,那怎么办,难道让我开发到这里,写了这么多的配置文件,突然之间就去换成ibatis. 其实大可不必,为了符合上述的编码规范,我们通常采用sql配置化的方式,将sql语句保存在配置文件中,需要调用或修改的时候,直接去读取配置就行了.

引用 南桥《呀,美利坚》读后记录

- 躲在街角的猫 - 南桥的博客
本不想写,最近觉得有点推销名人书籍的意味,嘿,可是欢喜,仍是愿意与朋友一起分享,也梳理自己的读书所得. 南桥的文章,读过不少,很欣赏他辛辣而幽默的笔下流淌出来的文字,即便是苦痛的事情,也能让你苦中作乐. 偶尔跟跟贴,常常推荐阅读,读很多篇后,看南桥在文最后发广告,自己觉得老是这样白读人家的好文章,也有点不好意思,再说了,仍是喜欢圈圈点点地看纸质书的,于是,就在当当网买了一本.

如何吸引用户的眼球之产品摄影

- 骁炜 - 牛博山寨 编辑推荐
电商网站用户体验的作用体现在两个方面,一个方面是吸引用户购买,另一个方面是使得购买流程变的顺畅. 很明显的,我们看到的某些用户体验糟糕的网站产品销量却很不错的部分原因就在于,他们在吸引用户购买这个环节做的不错,但是在改善购买流程方面有点不尽如人意. 好的,那吸引用户购买的环节包含了哪些内容?. 价格因素、促销因素、产品本身的价值、产品图片等等,他们共同形成了对用户的强大吸引力,诱惑他去买这个产品.

【引用】看完这组图,我的腿软了!

- Qing-Run - 湖南谭颂德的实名博客

引用】买车必看,别给奸商蒙了

- - jnehmkxmch的博客
本文引用自清风流水 《【引用】买车必看,别给奸商蒙了》. 趴下身看地下都没有机油点,底盘有没有油污. 新车电瓶没啥可多看的,只看接头有否腐蚀以及小窗是否绿色就行了. 注意电瓶接头一般是松的,开走前一定要JS拧紧. 第二步:拉出机油尺看机油颜色. 有人又要说,不都是0公里吗. 当然,“0公里”并不是指里程表上显示的数字绝对为“0”,一般认为,里程表在60公里以下的,都可以认为是“零公里”.

浅析如何吸引用户注意力

- - 所有文章 - UCD大社区
注意力是指人的心理活动指向和集中于某种事物的能力. 通常我们在地铁上拿着手机刷微博,很可能关注不到这一站上来了多少乘客或者旁边的乘客在谈论的内容,甚至还有可能坐过站. 我们操作电脑会打开N个窗口,我们在听着歌浏览某个新闻,同时也在关注来自QQ的消息. 研究人员曾招募了一批大学生做测试,观察他们日常无任务状态操作电脑/电视时的举动.

Weblogic 设置优先引用web项目的jar包

- - ITeye博客
在WEB-INF/weblogic.xml中进行如下配置:. 已有 0 人发表留言,猛击->> 这里<<-参与讨论. —软件人才免语言低担保 赴美带薪读研.

【引用】十分钟让你的父母多活10年

- - 湖南谭颂德的实名博客
                         十分钟让你的父母多活10年 .                                                                        编辑:鲍新宇. 十分钟却可以让你和你的父母都至少多活十年-------北大齐教授健康讲座笔录.