程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C >> C語言基礎知識 >> 基於C語言中段錯誤的問題詳解

基於C語言中段錯誤的問題詳解

編輯:C語言基礎知識

當我在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型變量 這個應該是一個棧地址 內核已經把它映射到一個實際的物理頁
你在這個基礎上小幅度上下偏移地址.相應的都應該有物理地址與之映射.
自然沒有問題.

以上都是自己的個人理解.可能還有不足的地方.
歡迎大家交流指教!

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved