浅析段错误和栈溢出
什么段
??我們在學(xué)習(xí)微機(jī)原理的時(shí)候就遇到過段,它代表在一個(gè)可執(zhí)行文件中各種的類型信息存放的地方。
??正文段:text用于存儲指令;
??數(shù)據(jù)段:data用于存儲已初始化的全局變量;
??bss段:用來存放程序中未初始化或者初始化為0的全局變量和靜態(tài)變量;
??堆棧段:stack和其他段一樣有著自己的大小,一旦越界同樣會爆段錯(cuò)誤。它是在運(yùn)行時(shí),程序動態(tài)創(chuàng)建的一個(gè)堆棧段,放著調(diào)用棧,保存著函數(shù)調(diào)用關(guān)系和局部變量。
造成段錯(cuò)誤的原因一般有如下三點(diǎn):
1.內(nèi)存訪問出錯(cuò)
??這類問題的典型代表就是數(shù)組越界、變量類型不一致等。
2.非法內(nèi)存訪問
??這類問題主要是程序試圖訪問內(nèi)核段內(nèi)存而產(chǎn)生的錯(cuò)誤。
3.棧溢出
??棧,就是用來描述函數(shù)之間的調(diào)用關(guān)系,它由多個(gè)棧幀組成,每個(gè)棧幀代表著對應(yīng)運(yùn)行的函數(shù)。棧溢出指的是對棧的使用超出的棧的大小。引起棧溢出的原因主要有二,一個(gè)是局部變量所占用的空間太大(解決方法:增大棧空間或者用動態(tài)分配,使用堆),二是函數(shù)的調(diào)用/遞歸次數(shù)太多或者無限調(diào)用。
典型的段錯(cuò)誤:
int main(void){ char*s ="hello world";*s ='H';}??在程序被裝載時(shí),系統(tǒng)把“hello world” 連同其它字符串和const類型數(shù)據(jù)放入到內(nèi)存的只讀區(qū)域。在執(zhí)行時(shí),一個(gè)變量s被設(shè)為指向該字符串的位置,當(dāng)再向該位置寫時(shí),就會產(chǎn)生段錯(cuò)誤。
int main() {int*ptr = NULL;*ptr =1; }??該代碼僅僅創(chuàng)建了一個(gè)空指針,沒有指向一個(gè)具體空間,當(dāng)賦值時(shí),找不到具體的空間,所以會產(chǎn)生段錯(cuò)誤。
int main(void) {int main();return 0; }??這個(gè)代碼是很明顯的無限遞歸,這會導(dǎo)致棧溢出,從而產(chǎn)生段錯(cuò)誤。
總結(jié)
- 上一篇: 最长升序子串1231
- 下一篇: Java8 Stream详解~归约(re