1. 前言
在上一篇文章中,介紹了JVM中垃圾回收的原理和算法。介紹了通過引用計(jì)數(shù)和對(duì)象可達(dá)性分析的算法來篩選出已經(jīng)沒有使用的對(duì)象,然后介紹了垃圾收集器中使用的三種收集算法:標(biāo)記-清除、標(biāo)記-整理、標(biāo)記-復(fù)制算法。
介紹完原理,在這篇文章中,我們將介紹當(dāng)前JVM中已經(jīng)實(shí)現(xiàn)的垃圾收集器,以及與收集器主題相關(guān)的一些內(nèi)容。
首先,我們將在上一篇文章中提到分代收集機(jī)制的基礎(chǔ)上,介紹下現(xiàn)代商業(yè)JVM中普遍采用的分代回收策略。然后,按照內(nèi)存分代劃分的維度介紹下當(dāng)前JVM中實(shí)現(xiàn)的收集器。最后,學(xué)習(xí)分析不同收集器的GC日志,然后結(jié)合日志分析,學(xué)習(xí)下不同情況下的對(duì)象分配策略。
2. 分代收集策略
我們知道,當(dāng)對(duì)象被創(chuàng)建的時(shí)候,就會(huì)給對(duì)象分配一塊內(nèi)存空間,而一旦對(duì)象的生命周期結(jié)束,我們就需要回收這塊內(nèi)存空間。但是,在一個(gè)應(yīng)用程序中,不同的對(duì)象存在的時(shí)間,或者說每個(gè)對(duì)象的生命周期都是不同的。
有些對(duì)象生命周期很短,比如Web應(yīng)用程序中的request對(duì)象,它的生命周期和請(qǐng)求是對(duì)應(yīng)的,當(dāng)請(qǐng)求完成以后,該request對(duì)象就結(jié)束了它的職責(zé),需要被收集器回收。有些對(duì)象的生命周期很長,比如一些全局的對(duì)象,可能會(huì)伴隨整個(gè)應(yīng)用程序的生命周期而存在。
在上圖中,橫軸表示對(duì)象的生命周期長短,豎軸表示對(duì)應(yīng)生命周期下的對(duì)象數(shù)量。觀察藍(lán)色的區(qū)域,我們可以看到大部分的對(duì)象的生命周期都很短,而生命周期長的對(duì)象,它們的數(shù)量占據(jù)了小部分。
考慮到不同生命周期的對(duì)象的分布情況,為了合理的處理不同生命周期的對(duì)象回收問題。現(xiàn)代JVM的對(duì)不同生命周期的對(duì)象進(jìn)行分類,對(duì)堆內(nèi)存區(qū)域進(jìn)行邏輯劃分。按照對(duì)象的存活時(shí)間長短,將內(nèi)存分為:年輕代、老年代和永久代(在Java8中去掉了永久代,以元數(shù)據(jù)空間代替)。這里我們主要關(guān)注年輕代和老年代的GC。
JVM提供了兩個(gè)參數(shù)來控制JVM堆的大?。?span style="margin: 0px; padding: 0px; font-size: 12px;">-XX:InitialHeapSize(-Xms)和-XX:MaxHeapSize(-Xmx)。JVM會(huì)根據(jù)應(yīng)用程序使用內(nèi)存的情況,動(dòng)態(tài)擴(kuò)展堆內(nèi)存的大小,上圖中的Virtual表示的區(qū)域,表示的就是可以擴(kuò)展的內(nèi)存空間。
比如,我們可以將JVM的堆內(nèi)存設(shè)置為256M,最大512M的大小,那么可以這么設(shè)置:-Xms256m -Xmx512m。如果將Xms的值和Xmx的值設(shè)置為相同,那么JVM將不能動(dòng)態(tài)擴(kuò)展堆內(nèi)存,它的初始堆內(nèi)存和最大堆內(nèi)存是相同的。