java的堆和棧 關(guān)于java堆棧的問(wèn)題,在編譯的時(shí)候入棧的順序是怎么樣的?
關(guān)于java堆棧的問(wèn)題,在編譯的時(shí)候入棧的順序是怎么樣的?事實(shí)上,JVM堆棧是一種以“堆棧幀”為單位的順序訪問(wèn)結(jié)構(gòu)。在JVM中,方法調(diào)用將在JVM堆棧上分配(put)和取消(put)相應(yīng)的堆棧幀。當(dāng)方
關(guān)于java堆棧的問(wèn)題,在編譯的時(shí)候入棧的順序是怎么樣的?
事實(shí)上,JVM堆棧是一種以“堆棧幀”為單位的順序訪問(wèn)結(jié)構(gòu)。在JVM中,方法調(diào)用將在JVM堆棧上分配(put)和取消(put)相應(yīng)的堆棧幀。當(dāng)方法離開(正常返回或異常)時(shí),撤消堆棧幀(即堆棧外)。
如何準(zhǔn)確理解Java中的堆與棧?
謝謝
!Java運(yùn)行時(shí)數(shù)據(jù)區(qū)有Java虛擬機(jī)棧和本地方法棧,用于執(zhí)行方法;堆用于存儲(chǔ)對(duì)象實(shí)例和數(shù)組。
Java虛擬機(jī)堆棧
執(zhí)行每個(gè)Java方法時(shí),將創(chuàng)建一個(gè)堆??蚣?。堆棧幀用于存儲(chǔ)局部變量列表、操作數(shù)堆棧、動(dòng)態(tài)鏈表和方法出口等信息。每個(gè)Java方法從調(diào)用開始到執(zhí)行完成的過(guò)程,對(duì)應(yīng)于Java虛擬機(jī)棧中棧幀從棧入口到棧出口的過(guò)程。根據(jù)Java虛擬機(jī)規(guī)范,如果線程請(qǐng)求的深度大于虛擬機(jī)允許的深度,則拋出StackOverflowerError異常;如果虛擬機(jī)可以動(dòng)態(tài)擴(kuò)展,但擴(kuò)展過(guò)程中內(nèi)存不足,則拋出outofmemoryerror異常。Java虛擬機(jī)棧的生命周期遵循線程,是線程的私有生命周期。
Local method stack
Java虛擬機(jī)堆棧用于執(zhí)行Java方法,而本地方法堆棧用于執(zhí)行本地方法。它的功能類似于Java虛擬機(jī)堆棧,還拋出StackOverflowerError異常和outofmemoryerror異常。本地方法堆棧的生存期也是線程跟隨和線程私有的。
Heap
堆是Java虛擬機(jī)中最大的內(nèi)存空間?;旧?,所有對(duì)象實(shí)例和數(shù)組都在這里分配內(nèi)存空間,這是所有線程共享的。在Java虛擬機(jī)規(guī)范中,堆可以位于物理上不連續(xù)的內(nèi)存空間中,只要它在邏輯上是連續(xù)的。當(dāng)堆中沒(méi)有足夠的內(nèi)存分配并且此時(shí)無(wú)法擴(kuò)展時(shí),將拋出outofmemoryerror異常。
請(qǐng)高手詳細(xì)的解答一下JAVA的堆,棧,方法區(qū),謝謝?
方法區(qū)域(非堆):它是所有線程共享的內(nèi)存區(qū)域,用于存儲(chǔ)類信息、常量、靜態(tài)變量、編譯器編譯的代碼以及虛擬機(jī)加載的其他數(shù)據(jù)。Java堆:它是虛擬機(jī)管理的最大內(nèi)存區(qū)域,也是所有線程共享的內(nèi)存區(qū)域。它是在虛擬機(jī)啟動(dòng)時(shí)創(chuàng)建的。這個(gè)內(nèi)存區(qū)域的唯一用途是存儲(chǔ)對(duì)象實(shí)例,幾乎所有的對(duì)象實(shí)例都分配內(nèi)存。Java堆是垃圾收集器管理的主要領(lǐng)域。Java虛擬機(jī)棧:線程是私有的,其生命周期與線程相同。在執(zhí)行每個(gè)方法時(shí),將同時(shí)創(chuàng)建一個(gè)堆棧幀來(lái)存儲(chǔ)局部變量表、操作數(shù)堆棧、動(dòng)態(tài)鏈接和方法出口等信息。每個(gè)方法都會(huì)被調(diào)用,直到執(zhí)行完成,這與虛擬機(jī)中從堆棧到堆棧輸出進(jìn)程的堆棧幀相對(duì)應(yīng)。
java遞歸改為循環(huán)后為什么不會(huì)導(dǎo)致棧內(nèi)存溢出?
我們知道,在編程中,如果您希望業(yè)務(wù)被重復(fù)執(zhí)行,通常有兩種方法來(lái)實(shí)現(xiàn)它:遞歸和循環(huán)。在實(shí)際的編碼過(guò)程中,我們不建議使用遞歸,而是建議使用循環(huán)。為什么?
事實(shí)上,不僅僅是Java,任何編程語(yǔ)言,如果遞歸寫入錯(cuò)誤,都可能導(dǎo)致內(nèi)存溢出
!學(xué)習(xí)過(guò)Java的朋友一定或多或少聽說(shuō)過(guò)并理解了堆棧內(nèi)存和堆內(nèi)存。程序運(yùn)行時(shí),計(jì)算機(jī)操作系統(tǒng)會(huì)給每個(gè)進(jìn)程分配堆內(nèi)存和堆棧內(nèi)存,分配的堆棧內(nèi)存有一個(gè)上限。一旦超過(guò)上限,就會(huì)導(dǎo)致內(nèi)存溢出。
為什么遞歸操作容易導(dǎo)致內(nèi)存溢出?主要原因如下:
在遞歸方法中,如果終止遞歸的條件寫得不正確,可能導(dǎo)致無(wú)限遞歸,最終導(dǎo)致內(nèi)存溢出;
即使遞歸方法和退出遞歸條件正常,如果遞歸深度太深(遞歸次數(shù)太多),也會(huì)導(dǎo)致堆棧內(nèi)存溢出!因?yàn)闂H霔3龅囊?guī)則是先入后出(先入后出),如果遞歸次數(shù)過(guò)多,就會(huì)導(dǎo)致只入不出棧,最后導(dǎo)致棧內(nèi)存溢出。
將遞歸寫入方式改為循環(huán)寫入方式的優(yōu)點(diǎn)是不會(huì)在短時(shí)間內(nèi)出現(xiàn)只進(jìn)不出棧的現(xiàn)象,避免了棧內(nèi)存溢出的現(xiàn)象。