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。

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