前言
- 對象的內存分配,大多數(shù)都是在堆上分配的放吩,但是有一些在JIT編譯后被拆散為標量類型并間接地棧上分配(即進行了優(yōu)化)智听。在堆上,對象會被分配在新生代(主要)和老年代(少數(shù))中,但是如果啟動了本地線程分配緩沖(-XX:+/-UseTLAB參數(shù)指定)到推,將按線程優(yōu)先在TLAB上分配考赛。
- 注:分配規(guī)則不是固定的,具體細節(jié)取決于當前使用的是哪一種垃圾收集器組合莉测,還有虛擬機中與內存相關的參數(shù)設置
- 下面講解幾條最普遍的內存分配規(guī)則颜骤。
1.對象優(yōu)先在Eden分配
- 堆上的大多數(shù)對象都是分配在新生代中,因為新生代采用的是復制算法捣卤,新生代中分為一個Eden區(qū)和兩個Survivor區(qū)忍抽,分配時會分配在Eden區(qū)和其中一個Survivor區(qū),當Eden+1個Survivor區(qū)沒有足夠空間分配時虛擬機將發(fā)起一次MinorGC董朝。此次收集會將Eden+1個Survivor區(qū)中不可達對象進行回收鸠项,并將存活的對象放入到另一個Survivor區(qū)中,然后再清理剛才使用的Eden+1個Survivor區(qū)子姜。注:如果Survivor區(qū)存不下Eden+1個Survivor區(qū)中存活下來的對象祟绊,就會采用分配擔保機制,將存活下來的對象轉移到老年代中哥捕。然后再清理剛才使用的Eden+1個Survivor區(qū)牧抽。
2.大對象直接進入老年代
- 所謂的大對象是指,需要大量連續(xù)內存空間的Java對象遥赚,最典型的大對象就是那種很長的字符串以及數(shù)組扬舒。
- 虛擬機提供了一個-XX:PretenureSizeThreshold參數(shù),令大于這個設置值的對象直接在老年代分配鸽捻。這樣做的目的是避免在Eden區(qū)及兩個Survivor區(qū)之間發(fā)生大量的內存復制(復習一下:新生代采用復制算法收集內存)呼巴。
3.長期存活對象進行老年代
- 既然虛擬機采用了分代收集的思想來管理內存泽腮,那么內存回收時就必須能識別哪些對象應放在新生代御蒲,哪些對象應放在老年代中。為了做到這點诊赊,虛擬機給每個對象定義了一個對象年齡(Age)計數(shù)器厚满。如果對象在Eden出生并經(jīng)過第一次Minor GC后仍然存活,并且能被Survivor容納的話碧磅,將被移動到Survivor空間中碘箍,并且對象年齡設為1。對象在Survivor區(qū)中每“熬過”一次Minor GC鲸郊,年齡就增加1歲丰榴,當它的年齡增加到一定程度(默認為15歲),就將會被晉升到老年代中秆撮。
- 對象晉升老年代的年齡閾值四濒,可以通過參數(shù)-XX:MaxTenuringThreshold設置。
4.動態(tài)對象年齡判定
- 為了能更好地適應不同程序的內存狀況,虛擬機并不是永遠地要求對象的年齡必須達到了MaxTenuringThreshold才能晉升老年代盗蟆,如果在Survivor空間中相同年齡所有對象大小的總和大于Survivor空間的一半戈二,年齡大于或等于該年齡的對象就可以直接進入老年代,無須等到MaxTenuringThreshold中要求的年齡喳资。
空間分配擔保
- 在發(fā)生Minor GC之前,虛擬機會先檢查老年代最大可用的連續(xù)空間是否大于新生代所有對象總空間,如果這個條件成立罢坝,那么Minor GC可以確保是安全的淋袖。如果不成立,則虛擬機會查看HandlePromotionFailure設置值是否允許擔保失敗节值。如果允許绒北,那么會繼續(xù)檢查老年代最大可用的連續(xù)空間是否大于歷次晉升到老年代對象的平均大小,如果大于察署,將嘗試著進行一次Minor GC闷游,盡管這次Minor GC是有風險的;如果小于贴汪,或者HandlePromotionFailure設置不允許冒險脐往,那這時也要改為進行一次Full GC。
- **注:Minor GC和Full GC有什么不一樣嗎扳埂?
新生代GC(Minor GC):指發(fā)生在新生代的垃圾收集動作业簿,因為Java對象大多都具備朝生夕滅的特性,所以Minor GC非常頻繁阳懂,一般回收速度也比較快梅尤。
老年代GC(Major GC/Full GC):指發(fā)生在老年代的GC,出現(xiàn)了Major GC岩调,經(jīng)常會伴隨至少一次的Minor GC(但非絕對的巷燥,在Parallel Scavenge收集器的收集策略里就有直接進行Major GC的策略選擇過程)。Major GC的速度一般會比Minor GC慢10倍以上号枕。
**