了解cms(concurrent mark
importnew注: 本文由新浪微博:@i李小武 投稿至importnew。感谢 @i李小武 ! 如果你希望分享好的原创文章或者译文,欢迎投稿到importnew。
1.总体介绍:
cms(concurrent mark-sweep)是以牺牲吞吐量为代价来获得最短回收停顿时间的垃圾回收器。对于要求服务器响应速度的应用上,这种垃圾回收器非常适合。在启动jvm参数加上-xx:+useconcmarksweepgc ,这个参数表示对于老年代的回收采用cms。cms采用的基础算法是:标记—清除。
2.cms过程:
- 初始标记(stw initial mark)
- 并发标记(concurrent marking)
- 并发预清理(concurrent precleaning)
- 重新标记(stw remark)
- 并发清理(concurrent sweeping)
- 并发重置(concurrent reset)
初始标记 :在这个阶段,需要虚拟机停顿正在执行的任务,官方的叫法stw(stop the word)。这个过程从垃圾回收的”根对象”开始,只扫描到能够和”根对象”直接关联的对象,并作标记。所以这个过程虽然暂停了整个jvm,但是很快就完成了。
并发标记 :这个阶段紧随初始标记阶段,在初始标记的基础上继续向下追溯标记。并发标记阶段,应用程序的线程和并发标记的线程并发执行,所以用户不会感受到停顿。
并发预清理 :并发预清理阶段仍然是并发的。在这个阶段,虚拟机查找在执行并发标记阶段新进入老年代的对象(可能会有一些对象从新生代晋升到老年代, 或者有一些对象被分配到老年代)。通过重新扫描,减少下一个阶段”重新标记”的工作,因为下一个阶段会stop the world。
重新标记 :这个阶段会暂停虚拟机,收集器线程扫描在cms堆中剩余的对象。扫描从”跟对象”开始向下追溯,并处理对象关联。
并发清理 :清理垃圾对象,这个阶段收集器线程和应用程序线程并发执行。
并发重置 :这个阶段,重置cms收集器的数据结构,等待下一次垃圾回收。
csm执行过程:
3.cms缺点
- cms回收器采用的基础算法是mark-sweep。所有cms不会整理、压缩堆空间。这样就会有一个问题:经过cms收集的堆会产生空间碎片。cms不对堆空间整理压缩节约了垃圾回收的停顿时间,但也带来的堆空间的浪费。为了解决堆空间浪费问题,cms回收器不再采用简单的指针指向一块可用堆空间来为下次对象分配使用。而是把一些未分配的空间汇总成一个列表,当jvm分配对象空间的时候,会搜索这个列表找到足够大的空间来hold住这个对象。
- 需要更多的cpu资源。从上面的图可以看到,为了让应用程序不停顿,cms线程和应用程序线程并发执行,这样就需要有更多的cpu,单纯靠线程切换是不靠谱的。并且,重新标记阶段,为空保证stw快速完成,也要用到更多的甚至所有的cpu资源。当然,多核多cpu也是未来的趋势!
- cms的另一个缺点是它需要更大的堆空间。因为cms标记阶段应用程序的线程还是在执行的,那么就会有堆空间继续分配的情况,为了保证在cms回收完堆之前还有空间分配给正在运行的应用程序,必须预留一部分空间。也就是说,cms不会在老年代满的时候才开始收集。相反,它会尝试更早的开始收集,已避免上面提到的情况:在回收完成之前,堆没有足够空间分配!默认当老年代使用68%的时候,cms就开始行动了。 – xx:cmsinitiatingoccupancyfraction =n 来设置这个阀值。
总得来说,cms回收器减少了回收的停顿时间,但是降低了堆空间的利用率。
4.啥时候用cms
如果你的应用程序对停顿比较敏感,并且在应用程序运行的时候可以提供更大的内存和更多的cpu(也就是硬件牛逼),那么使用cms来收集会给你带来好处。还有,如果在jvm中,有相对较多存活时间较长的对象(老年代比较大)会更适合使用cms。
作者:@i李小武
本文链接:http://www.importnew.com/2782.html
【如需转载,请联系作者 @i李小武 】
可能感兴趣的文章
- google play服务和oauth认证工具
- java的sizeof实现
- hibernate使用技巧
- 常见java面试题 – 第二部分:equals与==
- 2013 eclipse社区大奖最终获奖候选人名单