本文共 1631 字,大约阅读时间需要 5 分钟。
ELF(Executable Linkable Format)是Linux系统下的一种可执行可链接文件的格式,是COFF格式的变种。在Linux系统中包括了可重定位文件(.o文件),可执行文件(/bin/bash文件),共享目标文件(.so)和核心转储文件(core dump)。
先使用gcc -c SimpleTest.c生成目标文件SimpleTest.o
Ndx表示的是段在ELF文件中的下标,其中的4便是bss段,3则是.data段。
通过readelf我们可以看到,static_var 与global_init_var存放于.data段加起来正好为8对应着第一张图
而只有static_var2是存放在bss段中的,占4字节同时由于为初始化里面没有任何内容,而global_uninit_var却并没有存放于任何段中,只是一个未定义的“COMMON”符号。这涉及到了强符号和弱符号以及编译器的具体实现。
在C/C++语言的编译器中,默认函数名和初始化的全局变量是强符号,未初始化的全局变量为弱符号。由于C/C++语言编译器允许多个文件模块中出现同名的不同类型的弱符号,所以在编译期间,编译器并不能正确的知道这个弱符号所占的大小是多少,所以只能使用一个未定义的COMMON符号来代替。只有在链接期间,链接器在链接所有文件模块后,才能得出这些同名不同类型的弱符号所需占用的空间大小。举个例子,假设现在有两个文件模块a.o和b.o,其中a.o中有一个int类型名为global的未初始化全局变量,而b.o中有一个double类型也名为global的未初始化的全局变量。这样在编译过程中,由于编译器无法知道这个名为global的全局变量究竟应该占多大的内存空间,所以它使用了一个COMMEN来表示。在链接的过程中,链接器在对这两个模块进行链接后,知道了这个global变量最大需要分配8个字节的空间,此时global变量在bss段中所需的大小也就确定了。
参考:http://blog.csdn.net/phoenix500526/article/details/50099919