C語(yǔ)言中dlopen和dlsym的使用方式詳解
背景
為了是不同的邏輯解耦,一般會(huì)把各個(gè)業(yè)務(wù)封裝成動(dòng)態(tài)庫(kù),然后主邏輯去調(diào)用各個(gè)插件。這里有個(gè)問(wèn)題是,為什么以前我們都是通過(guò)include第三方的頭文件,然后通過(guò)連接器實(shí)現(xiàn),現(xiàn)在卻要利用dlopen呢?考慮以下情況,比如我們要用cublas這個(gè)庫(kù)的sgemm函數(shù)。
#include "cublas.h"
int main()
{
cublas:: Mat a, b;
cublas::sgemm(a,b);
}
我們知道cublas是英偉達(dá)提供的,人家每年都要更新動(dòng)態(tài)庫(kù)的,比如今年更新后,動(dòng)態(tài)庫(kù)的頭文件改了cublas_v2.h, 函數(shù)名改為sgemm_v2, 這樣一頓操作后,你不僅要升級(jí)庫(kù),也要修改已經(jīng)上線(xiàn)的代碼,假如這個(gè)sgemm函數(shù)在你源碼中出現(xiàn)了n多次,這將是一個(gè)災(zāi)難。但是通過(guò)下面的方式你就可以避免這個(gè)問(wèn)題:
// func.h
#include <stdio.h>
#include <stdlib.h>
#include <cublas_v2.h> // 如果你知道確切的函數(shù)返回信息,這個(gè)對(duì)應(yīng)下面的cublas_func可以自己寫(xiě)。
#include <dlfcn.h>
extern std::once_flag cublas_dso_flag;
extern void *cublas_dso_handle;
struct DynLoad__add {
template <typename... Args>
inline auto operator()(Args... args) -> DECLARE_TYPE(add, args...)
{
using cublas_func = decltype(::add(std::declval<Args>()...)) (*)(Args...);
std::call_once(cublas_dso_flag, []() {
cublas_dso_handle = dlopen("./libcublas.so", RTLD_LAZY);
});
static void *p_add = dlsym(cublas_dso_handle, "add");
return reinterpret_cast<cublas_func>(p_add)(args...);
}
};
extern DynLoad__add add;
// func.c
DynLoad__add add;
// main.cc
#include <stdio.h>
#include <stdlib.h>
#include "func.h"
int main()
{
add(2,7));
}
根據(jù)上面的代碼可以看到,只要你每次修改func.h文件的動(dòng)態(tài)庫(kù)路勁和函數(shù)名就可以了,其他用到的add函數(shù)根本不需要再去修改。真是很方便,上面的代碼參考paddle的源碼:paddle/fluid/platform/dynload/cublas.h
demo
生產(chǎn)動(dòng)態(tài)庫(kù)
int add(int a,int b)
{
return (a + b);
}
int sub(int a, int b)
{
return (a - b);
}
gcc -fPIC -shared caculate.c -o libcaculate.so
調(diào)用dlopen
#include <dlfcn.h> void *dlopen(const char *filename, int flag); char *dlerror(void); void *dlsym(void *handle, const char *symbol); int dlclose(void *handle);
dlopen是加載動(dòng)態(tài)鏈接庫(kù),flag可以設(shè)置不同的模式(RTLD_LAZY 暫緩決定,等有需要時(shí)再解出符號(hào), RTLD_NOW 立即決定,返回前解除所有未決定的符號(hào)。), dlopen可以返回動(dòng)態(tài)庫(kù)的句柄,dlsym是獲取動(dòng)態(tài)庫(kù)中的具體函數(shù)名或者變量名。dlopen是關(guān)閉動(dòng)態(tài)庫(kù)。
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>
typedef int (*FUNC)(int, int);
int main()
{
void *handle;
char *error;
FUNC func = NULL;
//打開(kāi)動(dòng)態(tài)鏈接庫(kù)
handle = dlopen("./libcaculate.so", RTLD_LAZY);
//獲取一個(gè)函數(shù)
*(void **) (&func) = dlsym(handle, "add");
printf("add: %d\n", (*func)(2,7));
//關(guān)閉動(dòng)態(tài)鏈接庫(kù)
dlclose(handle);
}
gcc -rdynamic -o main main.c -ldl
總結(jié)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C基礎(chǔ) redis緩存訪(fǎng)問(wèn)詳解
下面小編就為大家?guī)?lái)一篇C基礎(chǔ) redis緩存訪(fǎng)問(wèn)詳解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-06-06
C/C++獲取當(dāng)前時(shí)間的方法總結(jié)(最全)
這篇文章主要為大家整理了C/C++中獲取當(dāng)前時(shí)間的最全方法,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)和借鑒價(jià)值,需要的可以了解一下2023-03-03
CMake語(yǔ)法及CMakeList.txt簡(jiǎn)單使用小結(jié)
Cmake主要用于開(kāi)發(fā)跨平臺(tái)的C++項(xiàng)目,本文主要介紹了CMake語(yǔ)法及CMakeList.txt簡(jiǎn)單使用小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2022-05-05
C語(yǔ)言中字符的輸入輸出以及計(jì)算字符個(gè)數(shù)的方法詳解
這篇文章主要介紹了C語(yǔ)言中字符的輸入輸出以及計(jì)算字符個(gè)數(shù)的方法,是C語(yǔ)言入門(mén)學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-11-11
一篇文章帶你實(shí)現(xiàn)C語(yǔ)言中常用庫(kù)函數(shù)的模擬
這篇文章主要介紹了C語(yǔ)言中常用庫(kù)函數(shù)的模擬,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09
VisualStudio 使用Visual Leak Detector檢查內(nèi)存泄漏
這篇文章主要介紹了VisualStudio 使用Visual Leak Detector檢查內(nèi)存泄漏的相關(guān)資料,需要的朋友可以參考下2015-07-07
OpenCV實(shí)現(xiàn)簡(jiǎn)單錄屏功能
這篇文章主要為大家詳細(xì)介紹了OpenCV實(shí)現(xiàn)簡(jiǎn)單錄屏功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01

