多重定义的全局符号,链接器会如何链接的情况

多重定义的全局符号,链接器会如何链接的情况

以下只针对于gcc编译器,而且不同环境,不同编译器的情况可能不同。

假如说有多重定义的全局符号,链接器会如何链接呐?

在编译时,编译器向汇编器输出每个全局符号,根据它是强符号还是弱符号,把这个信息隐含地编码在可重定位目标文件的符号表李。

强符号:函数和已初始化的全局变量
弱符号:未初始化的全局变量。

根据强弱符号定义,Linux链接器使用下的规则来处理多重定义的符号名。

1.不允许有多个同名的强符号。

2.如果有一个强符号和多个弱符号同名,那么选择强符号。

3.如果有多个弱符号同名,那么从这些弱符号中任意选择

实例1:

1.规则1

a.c

int main()
{
     return 0;
}

b.c

int main()
{
     return 0;
}

编译结果:在这里插入图片描述

2.规则1

a.c

int x=2;
int main()
{
    return 0; 
 }

b.c

int x =1;

在这里插入图片描述

3.规则2

a.c

#include<stdio.h>
void f(void);
int x =2;
int main()
{
    f();
    printf("%d",x);
    return 0;
}

b.c

int x ;
void f()
{
     x=10;
}

运行结果
在这里插入图片描述

4.规则3

a.c

#include<stdio.h>
int x;
void fun(void);
int main()
{
    x =12;
    fun();
    printf("%d",x);
}

b.c

int x;
void fun()
{
    x=15;
}

运行结果
在这里插入图片描述

规则2和规则3会造成一些不易察觉的运行时的错误,我们一定要注意。

实例2

a.c

#include<stdio.h>
int x = 12;
int y = 13;
void fun(void);
int main()
{
    fun();
    printf("x= 0x%x y= 0x%x \n",x,y);
}

b.c

double x;
void fun()
{
     x =-0.0;
}

!!!此时b.c中的x=-0.0这一语句把a.c中 类型为int 的x和y都改了。

总结

当编译器在编译某个模块时,遇到一个弱全局符号,如x,它不知道其它模块是否也定义了该变量,如果定义了,它无法预测链接器使用哪一个x,所以编译器把x分配成COMMON,把决定权留给链接器,如果x被初始化0,那么它将是一个强符号,所以编译器直接将它成配成.bss。

所以我们在定义全局变量时,一定要初始化,避免使它成为弱符号!!!。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值