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