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