基于C語言中段錯誤的問題詳解
更新時間:2013年05月15日 10:46:49 作者:
本篇文章是對C語言中段錯誤的問題進行了詳細的分析介紹,需要的朋友參考下
當(dāng)我在linux下寫c語言的時候經(jīng)常會遇到段錯誤.
所以就來細究一下.
段錯誤或段違規(guī)(segmentation violation)
查看Expert C Programming(Peter Van Der Linden) Pg.156
解釋到段錯誤是由于內(nèi)存管理單元(MMU)的異常所致,
而該異常則通常是由于解除引用一個未初始化或非法的指針引起.
就是指針正在引用一個并不位于你的地址空間中的地址.
書中的例子
復(fù)制代碼 代碼如下:
int *p = 0;
*p = 17;
這里顯然 地址0 并不是你程序所在的地址空間 所能得到的
而我在試驗的時候 幾乎隨便給個地址 都是段錯誤
這也很正常,在運行之前是很難知道系統(tǒng)給你分配的地址空間的.
于是我這樣測試了一下
復(fù)制代碼 代碼如下:
int *p = 0;
int a = 7;
printf("a addr is %d\n",&a);
scanf("%ld",&p);
printf("%d",*p);
由于 變量a的地址肯定在系統(tǒng)給你的程序所分配的地址空間內(nèi)
所以你按照a的地址 給p賦值
或者小數(shù)目的向上下移4的整數(shù)倍 都是沒問題的
經(jīng)測試 并無段錯誤
分析了一下原因
在linux中,當(dāng)你malloc一段內(nèi)存的時候 只是拿到了 這段內(nèi)存的虛擬地址.而這段虛擬地址也名沒有實質(zhì)的映射到物理地址.
而只有當(dāng)你使用這段內(nèi)存的時候.系統(tǒng)會申請相應(yīng)頁表映射到相應(yīng)的物理地址.
而*p直接隨意指向一個虛擬地址 而這個虛擬地址并沒有實際的物理地址與之映射.
這時候解引用會在MMU發(fā)出異常,返回到linux就會給用戶報一個段錯誤.
而如果你定義1個int型變量 這個應(yīng)該是一個棧地址 內(nèi)核已經(jīng)把它映射到一個實際的物理頁
你在這個基礎(chǔ)上小幅度上下偏移地址.相應(yīng)的都應(yīng)該有物理地址與之映射.
自然沒有問題.
以上都是自己的個人理解.可能還有不足的地方.
歡迎大家交流指教!