库分为两种: 静态库和动态库.
对于静态库, 它的连接是静态连接, 其内容将复制到可执行文件.
对于动态库, 它的连接是动态连接, 其内容没有嵌入可执行文件, 细分为两种: 加载时连接和运行时连接.
本文用到的三个文件:
// shared_library.h
void foo(void);
// shared_library.c
#include "stdio.h"
void foo(void) {
printf("Hello, World!\n");
}
// shared_library_example.c
#include "shared_library.h"
int main(void) {
foo();
}
加载时连接
先看 CSAPP 的例子:
gcc -shared -fPIC -o libvector.so addvec.c multvec.c
gcc -o p2 main2.c ./libvector.so
连接时, ld 并未将动态库的代码和数据复制到可执行文件, 加载程序时, execve会使用动态连接器ld-linux.so将动态库的代码和数据加载到进程的虚拟内存.
再来看看我们的例子:
$ gcc -shared -fPIC -o libshared.so shared_library.c
$ gcc shared_library_example.c ./libshared.so
$ ./a.out
Hello, World!
改变动态库, 重新编译:
// shared_library.c
#include "stdio.h"
void foo(void) {
printf("hello, world!\n");
}
$ gcc -shared -f