C语言调用so动态库的两种方式

方式1:类似静态库的调用(使用头文件)

这种方式生成的程序会在启动时候就加载so动态库。

add.h

int add(int x, int y);

add.c

#include "add.h"

int add(int x, int y) {
    return (x + y);
}

main.c

#include <stdio.h>
#include "add.h"

int main()
{
    int sum = add(7, 8);
    printf("7+8 = %d\n", sum);

    return 0;
}

编译so,生成libadd.so。

gcc -shared -o libadd.so add.c

编译main,使用-L./指定add库在当前目录。

gcc -o main main.c -L./ -ladd

方式2:使用dlopen/dlsum动态加载动态库(不使用头文件)

这种方式生成的程序会在代码执行到指定行位置加载so动态库。

add.c

int add(int x, int y) {
    return (x + y);
}

main.c

#include <stdio.h>
#include <dlfcn.h>

int main()
{
    /*手动加载指定位置的so动态库*/
    void* handle = dlopen("./libadd.so", 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,生成libadd.so

gcc -shared -o libadd.so add.c

编译main,不需要指定libadd.so相关信息进行编译,执行时候会在指定目录加载so

gcc -o main main.c -ldl

两种调用方式总结

方式1使用头文件,所以可以直接调用头文件声明的函数。编译的时候指定了动态库位置和名称,程序启动时候系统就会自动加载相应位置的so动态库。
方式2没有头文件,编译的时候也不需要指定动态库信息。但是需要在程序中使用dlopen函数加载相应位置的so动态库,且要使用dlsym函数根据函数符号去查找此函数的地址。


BONUS: so动态库中调用so动态库

add.h

int add(int x, int y);

add.c

#include "add.h"

int add(int x, int y) {
    return (x + y);
}

sum.h

void printsum(int a, int b);

sum.c

#include "sum.h"
#include <stdio.h>
#include "add.h"

void printsum(int a, int b){
    int sum = add(a, b);
    printf("%d+%d = %d\n", a, b, sum);
}

main.c

#include "sum.h"

int main()
{
    printsum(1, 3);
    return 0;
}

编译libadd.so

gcc -shared -o libadd.so add.c

编译libsum.so,需要指定libadd.so信息

gcc -shared -o libsum.so sum.c -L. -ladd

编译main,仅需要指定libsum.so

gcc -o main main.c -L. -lsum

main运行的时候同时需要libsum.so 和 linadd.so。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章