O、什么是符号
符号是指代码中的变量与函数。
链接中的符号
一、分析
在外部符号和可引出符号的解析过程中,都有可能遇到符号重定义的问题。
当链接器发现符号重定义时会怎么处理吗?它会报错吗?
不要这么急着下结论。
本着“实践是检验真理的唯一标准”的原则,我们一起做几个实验。
二、实验材料
文件准备
head.h
#include <iostream>
using namespace std;
void myfun();
a.cpp
#include "head.h"
void myfun()
{
cout<<"myfun in a.cpp"<<endl;
}
b.cpp
#include "head.h"
void myfun()
{
cout<<"myfun in b.cpp"<<endl;
}
main.cpp
#include <iostream>
using namespace std;
#include "head.h"
int main()
{
myfun();
return 0;
}
文件加工
gcc/ld可以链接三种类型的可重定位目标文件,分别是目标文件(.oxx)、静态链接库(.a)和动态链接库(.so)。因此把a.c和b.c分别编译成不同的文件类型,来观察不同文件类型造成的符号重定义的链接结果。
(1)把a.cpp和b.cpp分别编译成a.oxx和b.oxx
```
(2)把a.cpp和b.cpp分别编译成liba.a和libb.a
(3)把a.cpp和b.cpp分别编译成liba.so和libb.so
```
三、实验过程及结果
实验序号 | a.cpp编译生成的文件类型 | b.cpp编译生成的文件类型 | 操作语句 | 链接结果(成功、失败) | 若成功,链接进去的是哪个文件,若失败,先链接进去的是哪个文件 |
---|---|---|---|---|---|
1 | a.oxx | b.oxx | g++ -o main main.cpp a.oxx b.oxx | 失败 | |
2 | a.oxx | libb.a | g++ -o main main.cpp a.oxx libb.a | 成功 | a.oxx |
3 | a.oxx | libb.so | g++ -o main main.cpp a.oxx -L. -lb | 成功 | a.oxx |
4 | liba.a | b.oxx | g++ -o main main.cpp liba.a b.oxx | 失败 | liba.a |
5 | liba.a | libb.a | g++ -o main main.cpp liba.a libb.a | 成功 | liba.a |
6 | liba.a | libb.so | g++ -o main main.cpp liba.a -L. -lb | 成功 | liba.a |
7 | liba.so | b.oxx | g++ -o main main.cpp -L. -la b.oxx | 成功 | b.oxx |
8 | liba.so | libb.a | g++ -o main main.cpp -L. -la libb.a | 成功 | liba.so |
9 | liba.so | libb.so | g++ -o main main.cpp -L. -la lb | 成功 | liba.so |
四、分析结果
(1)1 => 当一个符号在多个目标文件(.o)里同时出现时, LD报错. 提示符号多重定义.
普通目标文件的符号解析与重定义处理策略
(2)5, 6, 8, 9 => 当一个符号在多个静态库(.a)或者动态库(.so)里同时出现时, LD不报错, 以第一个遇到的为准.
(3)3, 7 => 当一个符号在目标文件(.o)和动态库(.so)里同时出现时,取目标文件(.o)里的符号
(4)2 =>
静态库的符号解析和重定义处理策略
有些情况下,链接器发现重定义后会提示链接错误,让我们及时地发现问题。可有些情况,链接器会悄悄地选择其中一个定义而忽略其它定义,这样就产生了难以定义的BUG。