JVM內(nèi)存組成結(jié)構(gòu)
JVM棧由堆、棧、本地方法棧穆刻、方法區(qū)等部分組成
1)堆
所有通過new創(chuàng)建的對象的內(nèi)存都在堆中分配喂饥,其大小可以通過-Xmx和-Xms來控制消约。堆被劃分為新生代和舊生代,新生代又被進一步劃分為Eden和Survivor區(qū)员帮,最后Survivor由From Space和To Space組成
新生代或粮。新建的對象都是用新生代分配內(nèi)存,Eden空間不足的時候集侯,會把存活的對象轉(zhuǎn)移到Survivor中被啼,新生代大小可以由-Xmn來控制,也可以用-XX:SurvivorRatio來控制Eden和Survivor的比例
舊生代棠枉。用于存放新生代中經(jīng)過多次垃圾回收仍然存活的對象
2)棧
每個線程執(zhí)行每個方法的時候都會在棧中申請一個棧幀浓体,每個棧幀包括局部變量區(qū)和操作數(shù)棧,用于存放此次方法調(diào)用過程中的臨時變量辈讶、參數(shù)和中間結(jié)果
3)本地方法棧
用于支持native方法的執(zhí)行命浴,存儲了每個native方法調(diào)用的狀態(tài)
4)方法區(qū)
存放了要加載的類信息、靜態(tài)變量贱除、final類型的常量生闲、屬性和方法信息。JVM用持久代(Permanet Generation)來存放方法區(qū)月幌,可通過-XX:PermSize和-XX:MaxPermSize來指定最小值和最大值
垃圾回收機制
JVM分別對新生代和舊生代采用不同的垃圾回收機制
新生代的GC:
新生代通常存活時間較短碍讯,因此基于Copying算法來進行回收,所謂Copying算法就是掃描出存活的對象扯躺,并復制到一塊新的完全未使用的空間中捉兴,對應于新生代,就是在Eden和From Space或To Space之間copy录语。新生代采用空閑指針的方式來控制GC觸發(fā)倍啥,指針保持最后一個分配的對象在新生代區(qū)間的位置,當有新的對象要分配內(nèi)存時澎埠,用于檢查空間是否足夠虽缕,不夠就觸發(fā)GC。當連續(xù)分配對象時蒲稳,對象會逐漸從eden到survivor氮趋,最后到舊生代,
用JavavisualVM來查看江耀,能明顯觀察到新生代滿了后凭峡,會把對象轉(zhuǎn)移到舊生代,然后清空繼續(xù)裝載决记,當舊生代也滿了后,就會報outofmemory的異常
在執(zhí)行機制上JVM提供了串行GC(Serial GC)倍踪、并行回收GC(Parallel Scavenge)和并行GC(ParNew)
1)串行GC
在整個掃描和復制過程采用單線程的方式來進行系宫,適用于單CPU索昂、新生代空間較小及對暫停時間要求不是非常高的應用上,是client級別默認的GC方式扩借,可以通過-XX:+UseSerialGC來強制指定
2)并行回收GC
在整個掃描和復制過程采用多線程的方式來進行椒惨,適用于多CPU、對暫停時間要求較短的應用上潮罪,是server級別默認采用的GC方式康谆,可用-XX:+UseParallelGC來強制指定,用-XX:ParallelGCThreads=4來指定線程數(shù)
3)并行GC
與舊生代的并發(fā)GC配合使用
舊生代的GC:
舊生代與新生代不同嫉到,對象存活的時間比較長沃暗,比較穩(wěn)定,因此采用標記(Mark)算法來進行回收何恶,所謂標記就是掃描出存活的對象孽锥,然后再進行回收未被標記的對象,回收后對用空出的空間要么進行合并细层,要么標記出來便于下次進行分配惜辑,總之就是要減少內(nèi)存碎片帶來的效率損耗。在執(zhí)行機制上JVM提供了串行GC(Serial MSC)疫赎、并行GC(parallel MSC)和并發(fā)GC(CMS)