本部分內容詳見《C專家編程》P119
查看可執行文件中的段
1.編譯“hello world”程序,在可執行文件中執行ls -l,得到文件的總體大小。運行size得到文件里各個段的大小。
2.增加一個全局的int[1000]數組聲明,重新進行編譯,再用上面的命令得到總體及各個段的大小,注意前后的區別。
3.現在,在數組的聲明中增加初始值(記住,C語言并不強迫對數組進行初始化時為每個元素提供初始值)。這將使數組從BSS段轉換到數據段。重復上面的測量,注意各個段前后大小的區別。
4.現在,在函數內聲明一個巨大的數組。然后再聲明一個巨大的局部數組,但這次加上初始值。重復上面的測量。定義于函數內部的局部數組存儲在可執行文件中嗎?有沒有初始化有什么不同嗎?
5.如果在調試狀態下編譯,文件和段的大小有沒有變化?是為了最大程度的優化嗎?
分析上面“編程挑戰”的結果,使自己確信:
- 數據段保存在目標文件中。
- BSS段不保存在目標文件中(除了記錄BSS段在運行時所需要的大小)。
- 文本段是最容易受優化措施影響的段。
- a.out文件的大小受調試狀態下編譯的影響,但段不受影響。
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后的數據如下所示:
[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借用了這個名字,至今依然沿用。