1. 程式人生 > >C語言呼叫so動態庫的兩種方式

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。