更新時(shí)間:2020-10-10 17:36:03 來(lái)源:動(dòng)力節(jié)點(diǎn) 瀏覽1745次
虛擬機(jī)作為Java基礎(chǔ)里面的重點(diǎn)內(nèi)容,在各大互聯(lián)網(wǎng)公司的面試題中頻頻出現(xiàn),本文總結(jié)了5道經(jīng)典JVM面試題,我是在學(xué)校學(xué)的java,雖然現(xiàn)在沒(méi)有從事相關(guān)的工作,但是確實(shí)參加過(guò)好幾次java面試,讓我記憶憂新的就是JVM面試題了,當(dāng)初這些題目雖然都不會(huì),但這些題目深深的印在腦海里,匯總并附上答案,希望能夠?qū)Υ蠹业那舐毱鸬揭稽c(diǎn)點(diǎn)幫助。
1.什么情況下會(huì)發(fā)生棧內(nèi)存溢出。
棧是線程私有的,他的生命周期與線程相同,每個(gè)方法在執(zhí)行的時(shí)候都會(huì)創(chuàng)建一個(gè)棧幀,用來(lái)存儲(chǔ)局部變量表,操作數(shù)棧,動(dòng)態(tài)鏈接,方法出口等信息。局部變量表又包含基本數(shù)據(jù)類型,對(duì)象引用類型。
如果線程請(qǐng)求的棧深度大于虛擬機(jī)所允許的最大深度,將拋出StackOverflowError異常,方法遞歸調(diào)用產(chǎn)生這種結(jié)果。
如果Java虛擬機(jī)棧可以動(dòng)態(tài)擴(kuò)展,并且擴(kuò)展的動(dòng)作已經(jīng)嘗試過(guò),但是無(wú)法申請(qǐng)到足夠的內(nèi)存去完成擴(kuò)展,或者在新建立線程的時(shí)候沒(méi)有足夠的內(nèi)存去創(chuàng)建對(duì)應(yīng)的虛擬機(jī)棧,那么Java虛擬機(jī)將拋出一個(gè)OutOfMemory 異常。(線程啟動(dòng)過(guò)多)
參數(shù) -Xss 去調(diào)整JVM棧的大小
2.詳解JVM內(nèi)存模型
JVM內(nèi)存結(jié)構(gòu)
程序計(jì)數(shù)器:當(dāng)前線程所執(zhí)行的字節(jié)碼的行號(hào)指示器,用于記錄正在執(zhí)行的虛擬機(jī)字節(jié)指令地址,線程私有。
Java虛擬棧:存放基本數(shù)據(jù)類型、對(duì)象的引用、方法出口等,線程私有。
Native方法棧:和虛擬棧相似,只不過(guò)它服務(wù)于Native方法,線程私有。
Java堆:java內(nèi)存最大的一塊,所有對(duì)象實(shí)例、數(shù)組都存放在java堆,GC回收的地方,線程共享。
方法區(qū):存放已被加載的類信息、常量、靜態(tài)變量、即時(shí)編譯器編譯后的代碼數(shù)據(jù)等。(即永久帶),回收目標(biāo)主要是常量池的回收和類型的卸載,各線程共享。
3.JVM內(nèi)存為什么要分成新生代,老年代,持久代。新生代中為什么要分為Eden和Survivor。
1)共享內(nèi)存區(qū)劃分
共享內(nèi)存區(qū) = 持久帶 + 堆
持久帶 = 方法區(qū) + 其他
Java堆 = 老年代 + 新生代
新生代 = Eden + S0 + S1
2)一些參數(shù)的配置
默認(rèn)的,新生代 ( Young ) 與老年代 ( Old ) 的比例的值為 1:2 ,可以通過(guò)參數(shù) –XX:NewRatio 配置。
默認(rèn)的,Edem : from : to = 8 : 1 : 1 ( 可以通過(guò)參數(shù) –XX:SurvivorRatio 來(lái)設(shè)定)
Survivor區(qū)中的對(duì)象被復(fù)制次數(shù)為15(對(duì)應(yīng)虛擬機(jī)參數(shù) -XX:+MaxTenuringThreshold)
3)為什么要分為Eden和Survivor?為什么要設(shè)置兩個(gè)Survivor區(qū)?
如果沒(méi)有Survivor,Eden區(qū)每進(jìn)行一次Minor GC,存活的對(duì)象就會(huì)被送到老年代。老年代很快被填滿,觸發(fā)Major GC.老年代的內(nèi)存空間遠(yuǎn)大于新生代,進(jìn)行一次Full GC消耗的時(shí)間比Minor GC長(zhǎng)得多,所以需要分為Eden和Survivor。
Survivor的存在意義,就是減少被送到老年代的對(duì)象,進(jìn)而減少Full GC的發(fā)生,Survivor的預(yù)篩選保證,只有經(jīng)歷16次Minor GC還能在新生代中存活的對(duì)象,才會(huì)被送到老年代。
設(shè)置兩個(gè)Survivor區(qū)最大的好處就是解決了碎片化,剛剛新建的對(duì)象在Eden中,經(jīng)歷一次Minor GC,Eden中的存活對(duì)象就會(huì)被移動(dòng)到第一塊survivor space S0,Eden被清空;等Eden區(qū)再滿了,就再觸發(fā)一次Minor GC,Eden和S0中的存活對(duì)象又會(huì)被復(fù)制送入第二塊survivor space S1(這個(gè)過(guò)程非常重要,因?yàn)檫@種復(fù)制算法保證了S1中來(lái)自S0和Eden兩部分的存活對(duì)象占用連續(xù)的內(nèi)存空間,避免了碎片化的發(fā)生)
4. JVM中一次完整的GC流程是怎樣的,對(duì)象如何晉升到老年代
Java堆 = 老年代 + 新生代
新生代 = Eden + S0 + S1
當(dāng) Eden 區(qū)的空間滿了, Java虛擬機(jī)會(huì)觸發(fā)一次 Minor GC,以收集新生代的垃圾,存活下來(lái)的對(duì)象,則會(huì)轉(zhuǎn)移到 Survivor區(qū)。
大對(duì)象(需要大量連續(xù)內(nèi)存空間的Java對(duì)象,如那種很長(zhǎng)的字符串)直接進(jìn)入老年態(tài);
如果對(duì)象在Eden出生,并經(jīng)過(guò)第一次Minor GC后仍然存活,并且被Survivor容納的話,年齡設(shè)為1,每熬過(guò)一次Minor GC,年齡+1,若年齡超過(guò)一定限制(15),則被晉升到老年態(tài)。即長(zhǎng)期存活的對(duì)象進(jìn)入老年態(tài)。
老年代滿了而無(wú)法容納更多的對(duì)象,Minor GC 之后通常就會(huì)進(jìn)行Full GC,F(xiàn)ull GC 清理整個(gè)內(nèi)存堆 – 包括年輕代和年老代。
Major GC 發(fā)生在老年代的GC,清理老年區(qū),經(jīng)常會(huì)伴隨至少一次Minor GC,比Minor GC慢10倍以上。
5.你知道哪幾種垃圾收集器,各自的優(yōu)缺點(diǎn),重點(diǎn)講下cms和G1,包括原理,流程,優(yōu)缺點(diǎn)。
1)幾種垃圾收集器:
Serial收集器: 單線程的收集器,收集垃圾時(shí),必須stop the world,使用復(fù)制算法。
ParNew收集器: Serial收集器的多線程版本,也需要stop the world,復(fù)制算法。
Parallel Scavenge收集器: 新生代收集器,復(fù)制算法的收集器,并發(fā)的多線程收集器,目標(biāo)是達(dá)到一個(gè)可控的吞吐量。如果虛擬機(jī)總共運(yùn)行100分鐘,其中垃圾花掉1分鐘,吞吐量就是99%。
Serial Old收集器: 是Serial收集器的老年代版本,單線程收集器,使用標(biāo)記整理算法。
Parallel Old收集器: 是Parallel Scavenge收集器的老年代版本,使用多線程,標(biāo)記-整理算法。
CMS(Concurrent Mark Sweep) 收集器: 是一種以獲得最短回收停頓時(shí)間為目標(biāo)的收集器,標(biāo)記清除算法,運(yùn)作過(guò)程:初始標(biāo)記,并發(fā)標(biāo)記,重新標(biāo)記,并發(fā)清除,收集結(jié)束會(huì)產(chǎn)生大量空間碎片。
G1收集器: 標(biāo)記整理算法實(shí)現(xiàn),運(yùn)作流程主要包括以下:初始標(biāo)記,并發(fā)標(biāo)記,最終標(biāo)記,篩選標(biāo)記。不會(huì)產(chǎn)生空間碎片,可以精確地控制停頓。
2)CMS收集器和G1收集器的區(qū)別:
CMS收集器是老年代的收集器,可以配合新生代的Serial和ParNew收集器一起使用;
G1收集器收集范圍是老年代和新生代,不需要結(jié)合其他收集器使用;
CMS收集器以最小的停頓時(shí)間為目標(biāo)的收集器;
G1收集器可預(yù)測(cè)垃圾回收的停頓時(shí)間
CMS收集器是使用“標(biāo)記-清除”算法進(jìn)行的垃圾回收,容易產(chǎn)生內(nèi)存碎片
G1收集器使用的是“標(biāo)記-整理”算法,進(jìn)行了空間整合,降低了內(nèi)存空間碎片。
以上就是提供給大家的5道經(jīng)典JVM面試題,在未來(lái)的求職之路上,假如遇到類似的題目我們也不至于毫無(wú)頭緒,當(dāng)然如果你想要快速提升自己,拿到offer,建議觀看學(xué)習(xí)本站的Java零基礎(chǔ)教程,里面還有更多的優(yōu)質(zhì)學(xué)習(xí)和面試資料等你來(lái)學(xué)哦。
0基礎(chǔ) 0學(xué)費(fèi) 15天面授
有基礎(chǔ) 直達(dá)就業(yè)
業(yè)余時(shí)間 高薪轉(zhuǎn)行
工作1~3年,加薪神器
工作3~5年,晉升架構(gòu)
提交申請(qǐng)后,顧問(wèn)老師會(huì)電話與您溝通安排學(xué)習(xí)