對(duì)象優(yōu)先在Eden分配
對(duì)象優(yōu)先在Eden分配
大多數(shù)情況下,對(duì)象在新生代Eden區(qū)中分配。當(dāng)Eden區(qū)沒(méi)有足夠空間進(jìn)行分配時(shí),虛擬機(jī)將發(fā)起一次Minor GC。
-Xms20M?-Xmx20M?-Xmn10M?-XX:+PrintGCDetails?-XX:SurvivorRatio=8
????@Test ????public?void?test3()?{ ????????int?_1MB=1024*1024; ????????byte[]allocation1,allocation2,allocation3,allocation4; ????????allocation1=new?byte[2*_1MB]; ????????allocation2=new?byte[2*_1MB]; ????????allocation3=new?byte[2*_1MB]; ????????allocation4=new?byte[4*_1MB];//出現(xiàn)一次Minor?GC ????}
----?IntelliJ?IDEA?coverage?runner?----? sampling?... include?patterns: exclude?patterns: [GC?[PSYoungGen:?8192K->1014K(9216K)]?8192K->2401K(19456K),?0.0030996?secs]?[Times:?user=0.00?sys=0.00,?real=0.03?secs]?Heap ?PSYoungGen??????total?9216K,?used?6745K?[0x00000000ff600000,?0x0000000100000000,?0x0000000100000000) ??eden?space?8192K,?69%?used?[0x00000000ff600000,0x00000000ffb98da0,0x00000000ffe00000) ??from?space?1024K,?99%?used?[0x00000000ffe00000,0x00000000ffefd878,0x00000000fff00000) ??to???space?1024K,?0%?used?[0x00000000fff00000,0x00000000fff00000,0x0000000100000000)
ParOldGen???????total?10240K,?used?1387K?[0x00000000fec00000,?0x00000000ff600000,?0x00000000ff600000) ??object?space?10240K,?13%?used?[0x00000000fec00000,0x00000000fed5aef0,0x00000000ff600000)
PSPermGen???????total?21248K,?used?5235K?[0x00000000f9a00000,?0x00000000faec0000,?0x00000000fec00000) ??object?space?21248K,?24%?used?[0x00000000f9a00000,0x00000000f9f1cd20,0x00000000faec0000)
[GC[DefNew:6651K->148K(9216K),0.0070106?secs]6651K->6292K(19456K),?0.0070426?secs]
[Times:user=0.00?sys=0.00,real=0.00?secs]?Heap
def?new?generation?total?9216K,used?4326K[0x029d0000,0x033d0000,0x033d0000)
eden?space?8192K,51%used[0x029d0000,0x02de4828,0x031d0000)
from?space?1024K,14%used[0x032d0000,0x032f5370,0x033d0000)
to?space?1024K,0%used[0x031d0000,0x031d0000,0x032d0000)
tenured?generation?total?10240K,used?6144K[0x033d0000,0x03dd0000,0x03dd0000)
the?space?10240K,60%used[0x033d0000,0x039d0030,0x039d0200,0x03dd0000)
compacting?perm?gen?total?12288K,used?2114K[0x03dd0000,0x049d0000,0x07dd0000)
the?space?12288K,17%used[0x03dd0000,0x03fe0998,0x03fe0a00,0x049d0000)
No?shared?spaces?configured.
嘗試分配3個(gè)2MB大小和1個(gè)4MB大小的對(duì)象,在運(yùn)行時(shí)通過(guò)-Xms20M、-Xmx20M、-Xmn10M這3個(gè)參數(shù)限制了Java堆大小為20MB,不可擴(kuò)展,其中10MB分配給新生代,剩下的10MB分配給老年代。-XX:SurvivorRatio=8決定了新生代中Eden區(qū)與一個(gè)Survivor區(qū)的空間比例是8:1,從輸出的結(jié)果看到,新生代總可用空間為9216KB(Eden區(qū)+1個(gè)Survivor區(qū)的總?cè)萘浚?br />
分配allocation4對(duì)象的語(yǔ)句時(shí)會(huì)發(fā)生一次Minor GC,這次GC的結(jié)果是新生代8192K變?yōu)?014K,而總內(nèi)存占用量則幾乎沒(méi)有減少(因?yàn)閍llocation1、allocation2、allocation3三個(gè)對(duì)象都是存活的,虛擬機(jī)幾乎沒(méi)有找到可回收的對(duì)象)。這次GC發(fā)生的原因是給allocation4分配內(nèi)存的時(shí)候,發(fā)現(xiàn)Eden已經(jīng)被占用了6MB,剩余空間已不足以分配allocation4所需的4MB內(nèi)存,因此發(fā)生Minor GC。GC期間虛擬機(jī)又發(fā)現(xiàn)已有的3個(gè)2MB大小的對(duì)象全部無(wú)法放入Survivor空間(Survivor空間只有1MB大?。?,所以只好通過(guò)分配擔(dān)保機(jī)制提前轉(zhuǎn)移到老年代去。
這次GC結(jié)束后,4MB的allocation4對(duì)象順利分配在Eden中,因此程序執(zhí)行完的結(jié)果是Eden占用4MB(被allocation4占用),Survivor空閑,老年代被占用6MB(被allocation1、allocation2、allocation3占用)。
新生代GC(Minor GC):指發(fā)生在新生代的垃圾收集動(dòng)作,因?yàn)镴ava對(duì)象大多都具備朝生夕滅的特性,所以Minor GC非常頻繁,一般回收速度也比較快。?
老年代GC(Major GC/Full GC):指發(fā)生在老年代的GC,出現(xiàn)了Major GC,經(jīng)常會(huì)伴隨至少一次的Minor GC(但非絕對(duì)的,在Parallel Scavenge收集器的收集策略里就有直接進(jìn)行Major GC的策略選擇過(guò)程)。Major GC的速度一般會(huì)比Minor GC慢10倍以上。
大對(duì)象直接進(jìn)入老年代
虛擬機(jī)提供了一個(gè)-XX:PretenureSizeThreshold參數(shù),令大于這個(gè)設(shè)置值的對(duì)象直接在老年代分配。這樣做的目的是避免在Eden區(qū)及兩個(gè)Survivor區(qū)之間發(fā)生大量的內(nèi)存復(fù)制(復(fù)習(xí)一下:新生代采用復(fù)制算法收集內(nèi)存)。
長(zhǎng)期存活的對(duì)象將進(jìn)入老年代
如果對(duì)象在Eden出生并經(jīng)過(guò)第一次Minor GC后仍然存活,并且能被Survivor容納的話,將被移動(dòng)到Survivor空間中,并且對(duì)象年齡設(shè)為1。對(duì)象在Survivor區(qū)中每“熬過(guò)”一次Minor GC,年齡就增加1歲,當(dāng)它的年齡增加到一定程度(默認(rèn)為15歲),就將會(huì)被晉升到老年代中。對(duì)象晉升老年代的年齡閾值,可以通過(guò)參數(shù)-XX:MaxTenuringThreshold設(shè)置。
動(dòng)態(tài)對(duì)象年齡判定
為了能更好地適應(yīng)不同程序的內(nèi)存狀況,虛擬機(jī)并不是永遠(yuǎn)地要求對(duì)象的年齡必須達(dá)到了MaxTenuringThreshold才能晉升老年代,如果在Survivor空間中相同年齡所有對(duì)象大小的總和大于Survivor空間的一半,年齡大于或等于該年齡的對(duì)象就可以直接進(jìn)入老年代,無(wú)須等到MaxTenuringThreshold中要求的年齡。
空間分配擔(dān)保
在發(fā)生Minor GC之前,虛擬機(jī)會(huì)先檢查老年代最大可用的連續(xù)空間是否大于新生代所有對(duì)象總空間,如果這個(gè)條件成立,那么Minor GC可以確保是安全的。如果不成立,則虛擬機(jī)會(huì)查看HandlePromotionFailure設(shè)置值是否允許擔(dān)保失敗。如果允許,那么會(huì)繼續(xù)檢查老年代最大可用的連續(xù)空間是否大于歷次晉升到老年代對(duì)象的平均大小,如果大于,將嘗試著進(jìn)行一次Minor GC,盡管這次Minor GC是有風(fēng)險(xiǎn)的;如果小于,或者HandlePromotionFailure設(shè)置不允許冒險(xiǎn),那這時(shí)也要改為進(jìn)行一次Full
GC。