由C語言代碼(文本文件)形成可執(zhí)行程序(二進制文件),需要經(jīng)過編譯-匯編-連接三個階段。編譯過程把C語言文本文件生成匯編程序,匯編過程把匯編程序形成二進制機器代碼,連接過程則將各個源文件生成的二進制機器代碼文件組合成一個文件。
程序中段的使用
下面用一個簡單的例子來說明C語言中變量和段的對應關(guān)系。C語言程序中的全局區(qū)(靜態(tài)區(qū)),實際對應著下述幾個段:RO Data; RW Data ; BSS Data.
一般來說,直接定義的全局變量在未初始化數(shù)據(jù)區(qū),如果該變量有初始化則是在已初始化數(shù)據(jù)區(qū)(RW Data),加上const則將放在只讀數(shù)據(jù)區(qū)。
const char ro[ ] = {“this is read only data”}; //只讀數(shù)據(jù)區(qū)
static char rw_1[ ] ={“this is global read write data”}; //已初始化讀寫數(shù)據(jù)段
char BSS_1[ 100]; //未初始化數(shù)據(jù)段
const char *ptrconst =”constant data”; //字符串放在只讀取數(shù)據(jù)段
int main()
{
short b; //在棧上,占用2個字節(jié)
char a[100]; //在棧上開辟100個字節(jié),它的值是其首地址
char s[ ]=”abcdefg”; //s在棧上,占用4個字節(jié),”abcdefg”本身放置在只讀數(shù)據(jù)存儲區(qū),占8個字節(jié)
char *p1; //p1在棧上,占用4個字節(jié)
char *p2=”123456″; //p2 在棧上,p2指向的內(nèi)容不能改,”123456″在只讀數(shù)據(jù)區(qū)
static char rw_2[ ]={“this is local read write data”};//局部已初始化讀寫數(shù)據(jù)段
static char BSS_2[100]; //局部未初始化數(shù)據(jù)段
static int c = 0; //全局(靜態(tài))初始化區(qū)
p1=(char *)malloc(10 * sizeof(char ));//分配內(nèi)存區(qū)域在堆區(qū)
strcpy(p1,”xxxx”);//”XXXX”放在只讀數(shù)據(jù)區(qū),占5個字節(jié)
free(p1);//使用free釋放p1所指向的內(nèi)存
return 0;
}
讀寫數(shù)據(jù)段包含了憶初始化的全局變量static char rw_1[ ]以及局部靜態(tài)變量static rw_2[ ].其差別在于編繹時,是在函數(shù)內(nèi)部使用的還是可以在整個文件中使用。對于rw_1[] 無論有無static 修飾,其都將被放置在讀寫數(shù)據(jù)區(qū),只是能否被其它文件引用與否。對于后者就不一樣了,它是局部靜態(tài)變量,放置在讀寫數(shù)據(jù)區(qū),如果沒static修飾,其意義完全改變,它將會是開辟在??臻g的局部變量,而不是靜態(tài)變量,在這里rw_1[],rw_2[]后沒具體數(shù)值,表示靜態(tài)區(qū)大小同后面字符串長度決定。
對于未初始化數(shù)據(jù)區(qū)BSS_1[100]與BSS_2[100],其區(qū)別在于前者是全局變量,在所有文件中都可以使用;后者是局部變量,只在函數(shù)內(nèi)部使用。未初始化數(shù)據(jù)段不設(shè)置后面的初始化數(shù)值,因此必須使用數(shù)值指定區(qū)域的大小,編繹器將根據(jù)大小設(shè)置BSS中需要增加的長度。
??臻g主要用于以下3數(shù)據(jù)的存儲:
1.函數(shù)內(nèi)部的動態(tài)變量
2.函數(shù)的參數(shù)
3.函數(shù)的返回值
??臻g是動態(tài)開辟與回收的。在函數(shù)調(diào)用過程中,如果函數(shù)調(diào)用的層次比較多,所需要的棧空間也逐漸加大,對于參數(shù)的傳遞和返回值,如果使用較大的結(jié)構(gòu)體,在使用的??臻g也會比較大。