对于C语言来说,编译器默认函数和初始化的全局变量为强符号,未初始化的全局变量为弱符号(C++并没有将未初始化的全局符号视为弱符号)。可以通过GCC的"__ attribute ((weak))"来定义任何一个强符号为弱符号,可以使用nm查看一个目标文件中的符号类型。 attribute__ ((weak))只对链接器有效,对编译器不起作用,编译器不区分强符号和弱符号,只要在一个源文件中定义两个相同的符号,不管它们是强是弱,都会报“重复定义”错误。
- 不同的目标文件中不能有同名的强符号,否则不能链接在一起,编译器报"重复定义"错误。
- 允许一个强符号和多个弱符号,但定义会选择强符号的。
- 当有多个弱符号相同时,默认的,链接器使用第一个找到的符号。
extern int ext;
int weak1;
int strong = 1;
int __attribute__((weak)) weak2 = 2;
int main()
{
return 0;
}
上面这段程序中,"weak1"和"weak2"是弱符号(C语言),"strong"和"main"是强符号,而"ext"既非强符号也非弱符号,因为它是一个外部变量的引用。
void __attribute__((weak)) f();
int main(void)
{
if (f)
f();
return 0;
}
声明了一个符号f(),属性为weak,但并不定义它,这样,链接器会将此未定义的weak symbol赋值为0,也就是说f()并没有真正被调用,试试看,去掉if条件,肯定core dump!
如果我们在另一个文件中定义函数f,与week.c一起编译链接,那么函数f就会正确的被调用。