方式1:类似静态库的调用(使用头文件)
这种方式生成的程序会在启动时候就加载so动态库。
int add(int x, int y);
#include ""
int add(int x, int y) {
return (x + y);
}
#include <>
#include ""
int main()
{
int sum = add(7, 8);
printf("7+8 = %d\n", sum);
return 0;
}
编译so,生成。
gcc -shared -o libadd.so add.c
编译main,使用-L./指定add库在当前目录。
gcc -o main main.c -L./ -ladd
方式2:使用dlopen/dlsum动态加载动态库(不使用头文件)
这种方式生成的程序会在代码执行到指定行位置加载so动态库。
int add(int x, int y) {
return (x + y);
}
#include <>
#include <>
int main()
{
/*手动加载指定位置的so动态库*/
void* handle = dlopen("./", RTLD_LAZY);
int (*add)(int a, int b);
/*根据动态链接库操作句柄与符号,返回符号对应的地址*/
add = dlsym(handle, "add");
int sum = add(7, 8);
printf("7+8 = %d\n", sum);
dlclose(handle);
return 0;
}
编译so,生成
gcc -shared -o libadd.so add.c
编译main,不需要指定相关信息进行编译,执行时候会在指定目录加载so
gcc -o main main.c -ldl
两种调用方式总结
方式1使用头文件,所以可以直接调用头文件声明的函数。编译的时候指定了动态库位置和名称,程序启动时候系统就会自动加载相应位置的so动态库。
方式2没有头文件,编译的时候也不需要指定动态库信息。但是需要在程序中使用dlopen函数加载相应位置的so动态库,且要使用dlsym函数根据函数符号去查找此函数的地址。
BONUS: so动态库中调用so动态库
int add(int x, int y);
#include ""
int add(int x, int y) {
return (x + y);
}
void printsum(int a, int b);
#include ""
#include <>
#include ""
void printsum(int a, int b){
int sum = add(a, b);
printf("%d+%d = %d\n", a, b, sum);
}
#include ""
int main()
{
printsum(1, 3);
return 0;
}
编译
gcc -shared -o libadd.so add.c
编译,需要指定信息
gcc -shared -o libsum.so sum.c -L. -ladd
编译main,仅需要指定
gcc -o main main.c -L. -lsum
main运行的时候同时需要 和 。