方式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。