淺談java.lang.StackOverflowError
StackOverflowError:棧溢出錯(cuò)誤镰惦,如果一個(gè)線程所需用到棧的大小>配置允許最大的棧大小拯杠,那么jvm就會(huì)拋出StackOverflow唇礁。查看一下程序可能是有死循環(huán)或遞歸調(diào)用所產(chǎn)生的,也可以增大JVM的內(nèi)存
棧的特點(diǎn)
1、棧,也叫棧內(nèi)存茬贵,是jvm的內(nèi)存模型之一,每當(dāng)啟動(dòng)一個(gè)新線程的時(shí)候移袍,jvm都會(huì)為它分配一個(gè)java棧解藻。jvm只會(huì)直接對(duì)java棧執(zhí)行兩種操作,以幀為單位的壓棧和出棧葡盗。
2螟左、棧存儲(chǔ)的內(nèi)容:方法內(nèi)的局部變量表、操作數(shù)觅够、動(dòng)態(tài)鏈接胶背、方法出口信息、其他等信息喘先。
- 1)局部變量表:保存函數(shù)的參數(shù)以及局部變量用的奄妨,局部變量表中的變量只在當(dāng)前函數(shù)調(diào)用中有效,當(dāng)函數(shù)調(diào)用結(jié)束后苹祟,隨著函數(shù)棧幀的銷毀,局部變量表也會(huì)隨之銷毀评雌。
- 2)操作數(shù):主要用于保存計(jì)算過程的中間結(jié)果树枫,同時(shí)作為計(jì)算過程中變量臨時(shí)的存儲(chǔ)空間。在概念模型中景东,兩個(gè)棧幀是相互獨(dú)立的砂轻,但是大多數(shù)虛擬機(jī)的實(shí)現(xiàn)都會(huì)進(jìn)行優(yōu)化,令兩個(gè)棧幀出現(xiàn)一部分重疊斤吐,令下面部分的操作數(shù)棧與上面部分的局部變量表重疊在一塊搔涝,這樣在方法調(diào)用的時(shí)候可以共用一部分?jǐn)?shù)據(jù)厨喂,無需進(jìn)行額外的參數(shù)復(fù)制傳遞。
- 3)動(dòng)態(tài)鏈接:每個(gè)棧幀都包含一個(gè)指向運(yùn)行時(shí)常量池中該棧幀所屬方法的引用庄呈,持有這個(gè)引用是為了支持方法調(diào)用過程中的動(dòng)態(tài)連接蜕煌。在Class文件的常量池中存有大量的 符號(hào)引用,字節(jié)碼中的方法調(diào)用指令就以常量池中指向方法的符號(hào)引用為參數(shù)诬留。這些符號(hào)引用一部分會(huì)在類加載階段或第一次使用的時(shí)候轉(zhuǎn)化為直接引用斜纪,這種轉(zhuǎn)化 稱為靜態(tài)解析。另外一部分將在每一次的運(yùn)行期期間轉(zhuǎn)化為直接引用文兑,這部分稱為動(dòng)態(tài)連接盒刚。
- 4)方法出口信息:在方法退出之前,都需要返回到方法被調(diào)用的位置绿贞,程序才能繼續(xù)執(zhí)行因块,方法返回時(shí)可能需要在棧幀中保存一些信息,用來恢復(fù)它的上層方法的執(zhí)行狀態(tài)籍铁,方法出口信息獲取分為正常退出和異常退出涡上。正常退出通過pc計(jì)數(shù)器的值獲取,異常退出通過異常處理器表確定返回地址寨辩。
- 5)附加信息:虛擬機(jī)規(guī)范中允許具體的虛擬機(jī)實(shí)現(xiàn)增加一些規(guī)范中沒有描述的信息到棧幀中吓懈,這部分信息取決于虛擬機(jī)的實(shí)現(xiàn)。
3靡狞、棧的生命周期:隨著線程的創(chuàng)建而創(chuàng)建耻警,線程的結(jié)束而消亡,釋放內(nèi)存甸怕,所以棧內(nèi)存是私有的
4甘穿、棧的存儲(chǔ)方式:棧內(nèi)存以棧幀(Stack Frame)為單位存儲(chǔ),棧幀是一個(gè)內(nèi)存區(qū)塊梢杭,是一個(gè)有關(guān)方法和運(yùn)行期數(shù)據(jù)的數(shù)據(jù)集温兼。當(dāng)一個(gè)方法M1被調(diào)用的時(shí)候,就會(huì)產(chǎn)生一個(gè)棧幀S1武契,并被壓入到棧中募判,M1方法又調(diào)用了M2方法,這個(gè)時(shí)候又產(chǎn)生棧幀S2也被壓入棧咒唆,M2方法執(zhí)行完畢后届垫,S2棧幀先出棧,S1棧幀再出棧全释,遵循“先進(jìn)后出”原則装处。
出現(xiàn)StackOverflowError的原因分析
一般出現(xiàn)這個(gè)問題是因?yàn)槌绦蚶镉兴姥h(huán)或遞歸調(diào)用所產(chǎn)生的。
如:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//死循環(huán)
int i=0;
while (true){
i++;
Log.i("ruxing","i="+i);
}
}
//遞歸
private void add(int i){
i++;
Log.i("ruxing","i="+i);
add(i);
}
}
以遞歸為例浸船,詳解程序:
1)啟動(dòng)MainActivity妄迁,會(huì)創(chuàng)建一個(gè)線程寝蹈,同時(shí)創(chuàng)建一個(gè)棧內(nèi)存。
2)調(diào)用add()方法的時(shí)候登淘,會(huì)對(duì)add()方法進(jìn)行壓棧操作箫老,將add()運(yùn)行期數(shù)據(jù)的數(shù)據(jù)集保存到棧幀中。
3)add()遞歸調(diào)用時(shí)形帮,都會(huì)產(chǎn)生一個(gè)新的棧幀區(qū)塊槽惫,這是就會(huì)連續(xù)的產(chǎn)生新的棧幀區(qū)塊。
4)當(dāng)棧內(nèi)存超過系統(tǒng)配置的棧內(nèi)存辩撑,就會(huì)出現(xiàn)java.lang.StackOverflowError異常界斜。