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