注:博客中内容主要来自《狄泰软件学院》,博客仅当私人笔记使用。
测试环境:Ubuntu 10.10
GCC版本:4.4.5
一、问题...
工程中的每个C语言源文件被编译后生产目标文件,这些目标文件如何生存最终的可执行程序?
二、链接器的意义
链接器的主要作用是把各个模块之间相互引用的部分处理好,使得各个模块之间能够正确的衔接。
注:这样就支持了多种开发语言混合开发项目。
三、模块链接
1)静态链接
- 由链接器在链接时将库的内容直接加入到可执行程序中。
链接时就将库加载到可执行程序(运行程序前就已经被加载完毕)。
2)linux下静态库的创建和使用
- 编译静态库源码:gcc -c lib.c -o lib.o
- 生成静态库文件:ar -q lib.a lib.o //ar:打包,lib是打包
//创建静态库lib.a
- 使用静态库编译:gcc main.c lib.a -o main.out
编程实验
静态链接示例
20-1.c
#include <stdio.h>
extern char* name();
extern int add(int a, int b);
int main()
{
printf("Name: %s\n", name());
printf("Result: %d\n", add(2, 3));
return 0;
}
slib.c
char* name()
{
return "Static Lib";
}
int add(int a, int b)
{
return a + b;
}
操作:
(1)编译静态库源码:gcc -c slib.c -o slib.o
(2)生成静态库文件:ar -q slib.a slib.o
ar: creating slib.a
(3)使用静态库:gcc 20-1.c slib.a -o 20-1.out
(4)运行结果:
Name:Static Lib
Result:5
3)动态链接
- 可执行程序在运行时才动态加载库进行链接
- 库的内容不会进入可执行程序当中
4)linux下动态库的创建和使用
- 编译动态库源码: gcc -shared dilb.c -o dlib.so
//创建动态库dlib.so
- 使用动态库编译: gcc main.c -ldl -o main.out
- 关键系统调用
*dlopen:打开动态库文件
*dlsym:查找动态库中的函数并返回调用地址
*dlclose:关闭动态库文件
编程实验
动态连接示例
20-2.c
#include <stdio.h>
#include <dlfcn.h>
int main()
{
//dlib.so动态库,linux C会详解这个函数
void* pdlib = dlopen("./dlib.so", RTLD_LAZY);
char* (*pname)(); //声明函数指针
int (*padd)(int, int);
if( pdlib != NULL )
{
pname = dlsym(pdlib, "name"); //获取函数地址
padd = dlsym(pdlib, "add");
//判断函数地址
if( (pname != NULL) && (padd != NULL) )
{
printf("Name: %s\n", pname()); //打印函数返回值
printf("Result: %d\n", padd(2, 3));
}
dlclose(pdlib); //关闭动态库
}
else
{
printf("Cannot open lib ...\n");
}
return 0;
}
dlib.c
char* name()
{
return "Dynamic Lib";
}
int add(int a, int b)
{
return a + b;
}
操作:
1) 演示使用动态库
创建动态库:gcc -shared dlib.c -o dlib.so
使用动态库编译:gcc 20-2.c -ldl -o 20-2.out
运行程序:
Name: Dynamic Lib
Result: 5
2) 删除动态库:dlib.so。运行程序:
Cannot open lib ...
分析:
打开库文件失败。找不到库文件。
动态链接程序运行对库的使用是在程序运行时调用库文件。
小结
1)链接是指将目标文件最终链接为可执行程序
2)根据链接方式的不同,链接过程可以分为:
- 静态链接:目标文件直接链接进入可执行程序
静态链接适合小程序
- 动态链接:在程序启动后才动态加载目标文件
程序更新时,只需要把某个动态库替换掉,从服务器下载下来,更新程序重启就完成升级。
使用静态链接还是动态链接取决于客户需求。