本部分內(nèi)容詳見《C專家編程》P119
查看可執(zhí)行文件中的段
1.編譯“hello world”程序,在可執(zhí)行文件中執(zhí)行l(wèi)s -l,得到文件的總體大小。運行size得到文件里各個段的大小。
2.增加一個全局的int[1000]數(shù)組聲明,重新進行編譯,再用上面的命令得到總體及各個段的大小,注意前后的區(qū)別。
3.現(xiàn)在,在數(shù)組的聲明中增加初始值(記住,C語言并不強迫對數(shù)組進行初始化時為每個元素提供初始值)。這將使數(shù)組從BSS段轉(zhuǎn)換到數(shù)據(jù)段。重復上面的測量,注意各個段前后大小的區(qū)別。
4.現(xiàn)在,在函數(shù)內(nèi)聲明一個巨大的數(shù)組。然后再聲明一個巨大的局部數(shù)組,但這次加上初始值。重復上面的測量。定義于函數(shù)內(nèi)部的局部數(shù)組存儲在可執(zhí)行文件中嗎?有沒有初始化有什么不同嗎?
5.如果在調(diào)試狀態(tài)下編譯,文件和段的大小有沒有變化?是為了最大程度的優(yōu)化嗎?
分析上面“編程挑戰(zhàn)”的結(jié)果,使自己確信:
- 數(shù)據(jù)段保存在目標文件中。
- BSS段不保存在目標文件中(除了記錄BSS段在運行時所需要的大小)。
- 文本段是最容易受優(yōu)化措施影響的段。
- a.out文件的大小受調(diào)試狀態(tài)下編譯的影響,但段不受影響。
helloworld.c的文件最終如下:
#include <stdio.h>
int global_array[1000] = {100, 101, 102};
void SayGoodBye(void);
int main(void)
{
printf("hello Linux world");
int main_array[1000] = {10, 20, 30, 40};
SayGoodBye();
return 0;
}
void SayGoodBye(void)
{
int goodbye_array[1000] = {1024, 2048, 4096};
printf("good bye user!");
}
以下命令按照文中步驟依次修改上文代碼(過程不贅述),運行size后的數(shù)據(jù)如下所示:
[volnet@GoCool c]$ echo "size helloworld1"; size helloworld1; echo "size helloworld2"; size helloworld2; echo "size helloworld3"; size helloworld3; echo "size helloworld4"; size helloworld4;
size helloworld1
text data bss dec hex filename
1015 252 8 1275 4fb helloworld1
size helloworld2
text data bss dec hex filename
1015 252 4032 5299 14b3 helloworld2
size helloworld3
text data bss dec hex filename
1015 4280 8 5303 14b7 helloworld3
size helloworld4
text data bss dec hex filename
1270 4280 8 5558 15b6 helloworld4
注:
BSS的全稱是:Block Started by Symbol(由符號開始的塊),它是舊式IBM704匯編程序的一個偽指令,UNIX借用了這個名字,至今依然沿用。