动态链接共享库
静态库有一些明显的缺陷,它与所有的软件一样,需要定期维护和更新。如果应用程序员想要使用一个库的最新版本,他们必须以某种方式了解到该库的更新情况,然后显示的将他们的程序与更新了的库重新链接。
其次,几乎每一个C程序都使用了标准I/O函数,运行时这些函数的代码会被复制到每个运行进程的文本段中,在运行上百个进程时将是对内存的极大浪费。
而共享库(dynamic object共享目标)就能很好的解决这个问题,共享库这个目标模块在运行或加载时,可以加载到任意的内存地址,并和一个在内存中的程序链接起来。这个过程被称为动态链接。
特点:
- 所有引用该库的可执行文件共享这个.so文件中的代码和数据。
- 在内存中一个共享库的.text节的一个副本可以被不同的正在运行的进程共享。
/* addvec.c */
/* $begin addvec */
int addcnt = 0;
void addvec(int *x, int *y,
int *z, int n)
{
int i;
addcnt++;
for (i = 0; i < n; i++)
z[i] = x[i] + y[i];
}
/* $end addvec */
/* multvec.c */
/* $begin multvec */
int multcnt = 0;
void multvec(int *x, int *y,
int *z, int n)
{
int i;
multcnt++;
for (i = 0; i < n; i++)
z[i] = x[i] * y[i];
}
/* $end multvec */
/* main2.c */
/* $begin main2 */
#include <stdio.h>
#include "vector.h"
int x[2] = {1, 2};
int y[2] = {3, 4};
int z[2];
int main()
{
addvec(x, y, z, 2);
printf("z = [%d %d]\n", z[0], z[1]);
return 0;
}
/* $end main2 */
/*vector.h*/
/* prototypes for libvector */
void addvec(int *x, int *y, int *z, int n);
void multvec(int *x, int *y, int *z, int n);
int getcount();
创建这个向量例程的共享库libvector.so:gcc -shared -fpic -o libvector.so addvec.c multvec.c
-fpic指示编译器生成位置无关代码,-shared选项指示链接器创建一个共享的目标文件。
创建库后,将他与main.o链接:gcc -o prog2l main2.c ./libvector.so
创建可执行文件prog2l时只是创建部分链接的可执行文件,链接器复制了一些重定位和符号表信息。
(./prog2l:加载器将可执行文件的代码和数据从磁盘复制到内存并运行,这个过程叫加载。)
prog2l文件在运行时可以和libvector.so链接,在程序加载时,动态完成链接过程。此时,没有任何libvector.so的代码和数据节真的被复制到可执行文件prog2l中,而是运行时会解析对libvector.so中代码和数据的引用。