GCC 强符号与弱符号
1)编译器默认函数和初始化了的全局变量为强符号,未初始化的全局变量为弱符号
2)强符号和弱符号都是针对定义来说的,不是针对符号的引用
3)使用__attribute__((weak))定义弱符号
extern int global_ext;
int global_weak1;
int global_strong = 1;
int __attribute__((weak)) global_weak2 = 2;
int func1(void)
{
return 0;
}
int __attribute__((weak)) func2(void)
{
return 0;
}
"global_weak1" "global_weak2" 和"func2" 是弱符号,"global_strong"和"func1"是强符号,而"global_ext"既非强符号也非弱符号,因为它是一个外部变量的引用。
链接器会按如下规则处理与选择被多次定义的全局符号:
规则1:不允许强符号被多次定义(即不同的目标文件中不能有同名的强符号,如果有多个强符号定义,则链接器报符号重复定义错误。
规则2:如果一个符号在某个目标文件中是强符号,在其他文件中都是弱符号,那么选择强符号。
规则3:如果一个符号在所有目标文件中都是弱符号,那么选择其中占用空间最大的一个。比如目标文件A定义全局变量global为int型,占4个字节;目标文件B定义global为char型,占1个字节,那么目标文件A和B链接后,符号global占4个字节(尽量不要使用多个不同类型的弱符号,否则容易导致很难发现的程序错误, 尤其定义不同类型数组时)