申明: 正如题如示,本篇讲的是Linux下是静态库与共享库,而Window下的动态链接库详细情况可见这篇文章:windows动态链接库 DLL 浅析。虽然原理,思想差不多,但是细节却各有不同。
一、静态库
// addvec.c void addvec(int* x, int* y, int*z, int n) { int i=0; for(; i< n;++i) z[i] = x[i] + y[i]; }
// multvec.c void multvec(int*x, int* y, int* z, int n) { int i = 0; for(; i < n; ++i) z[i] = x[i] * y[i]; }
/* main2.c */ #include <stdio.h> int x[2] = {1, 2}; int y[2] = {3, 4}; int z[2]={0}; int main() { addvec(x, y, z, 2); printf("z = [%d %d]\n", z[0], z[1]); return 0; }
#include<dlfcn.h> /* 加载和链接共享库 filename filename:共享库的名字 flag有:RTLD_LAZY, RTLD_NOW,二者均可以和RTLD_GLOBAL表示取或 */ void *dlopen(const char *filename, int flag); // 若成功则返回执行句柄的指针,否则返回NULL /*根据共享库操作句柄与符号,返回符号对应的地址 handle:共享库操作句柄 symbol:需要引用的符号名字 */ void *dlsym(void *handle, char *symbol); // 若成功则返回执行符号的指针(即地址),若出错则返回NULL /* 如果没有程序正在使用这个共享库,卸载该共享库 */ int dlclose(void *handle); // 若卸载成功,则返回0,否则返回-1 /* 捕捉最近发生的错误 */ const char *dlerror(void); // 若前面对dlopen,dlsym或dlclose调用失败,则返回错误消息,否则返回NULL
例子:
#include <stdio.h> #include <stdlib.h> #include <dlfcn.h> int x[2] = {1, 2}; int y[2] = {3, 4}; int z[2] ={0}; int main() { void *handle; void (*addvec)(int *, int *, int *,int); char *error; handle = dlopen("./libvector.so", RTLD_LAZY); if(!handle){ fprintf(stderr, "%s\n", dlerror()); exit(1); } addvec = dlsym(handle, "addvec"); if((error = dlerror()) != NULL){ fprintf(stderr, "%s\n", dlerror()); exit(1); } addvec(x, y, z, 2); printf("z = [%d %d]\n", z[0], z[1]); if(dlclose(handle) < 0){ fprintf(stderr, "%s\n", dlerror()); exit(1); } return 0; }
运行结果: -ldl参数:表示生成的对象模块需要用到共享库
Referebces:
1.《深入理解计算机系统》第7章:链接 P448-P479
2. 静态库、共享库、动态库的创建和使用 :http://bbs.chinaunix.net/thread-2037617-1-1.html
3. Linux 动态库剖析:http://www.ibm.com/developerworks/cn/linux/l-dynamic-libraries/