詳解 QT 源碼之 QLibrary 跨平臺調(diào)用動態(tài)庫實現(xiàn)
詳解 QT 源碼之 QLibrary 跨平臺調(diào)用動態(tài)庫實現(xiàn)是本文要講解的內(nèi)容,在不同同臺上動態(tài)庫的使用,先來看內(nèi)容。
1、win下動態(tài)庫調(diào)用有關(guān)的函數(shù)包括:
(1)LoadLibrary,裝載動態(tài)庫。
(2)GetProcAddress,獲取要引入的函數(shù),將符號名或標(biāo)識號轉(zhuǎn)換為DLL內(nèi)部地址。
(3)FreeLibrary,釋放動態(tài)鏈接庫。
2、unix上與動態(tài)庫調(diào)用有關(guān)的函數(shù)包括:
(1)_打開動態(tài)鏈接庫:dlopen,函數(shù)原型void *dlopen (const char *filename, int flag);
dlopen用于打開指定名字(filename)的動態(tài)鏈接庫,并返回操作句柄。
(2)取函數(shù)執(zhí)行地址:dlsym,函數(shù)原型為: void *dlsym(void *handle, char *symbol);
dlsym根據(jù)動態(tài)鏈接庫操作句柄(handle)與符號(symbol),返回符號對應(yīng)的函數(shù)的執(zhí)行代碼地址。
(3)關(guān)閉動態(tài)鏈接庫:dlclose,函數(shù)原型為: int dlclose (void *handle);
dlclose用于關(guān)閉指定句柄的動態(tài)鏈接庫,只有當(dāng)此動態(tài)鏈接庫的使用計數(shù)為0時,才會真正被系統(tǒng)卸載。
(4)動態(tài)庫錯誤函數(shù):dlerror,函數(shù)原型為: const char *dlerror(void); 當(dāng)動態(tài)鏈接庫操作函數(shù)執(zhí)行失敗時,dlerror可以返回出錯信息,返回值為NULL時表示操作函數(shù)執(zhí)行成功。
我們來分析下Qt的源代碼,看看Qt是如何封裝這兩種不同的調(diào)用動態(tài)庫的方法。
下面是我用vc編寫的一個動態(tài)庫中的函數(shù)add:
- extern "C" __declspec(dllexport) int __stdcall add(int a,int b)
- {
- return a+b;
- }
下面我就用QLibrary來調(diào)用一下:
- QLibrary lib("QtDllTest.dll");
- if (lib.load())
- {
- typedef int(*AddFunction)(int a,int b);
- AddFunction Add=(AddFunction)lib.resolve("add");
- if (!Add)
- {
- cout<<"failed"<<endl;
- }
- else
- {
- int m;
- m=Add(1,1); //來個計算1+1
- cout<<"result:"<<m<<endl;
- }
- lib.unload();
- }
- else
- {
- cout<<"failed"<<endl;
- }
首先將目錄切換到QTDIR\src\corelib\plugin,這里面就是QLibrary實現(xiàn)的源代碼,打開qlibrary_p.h(熟悉了Qt的常用手法,就知道,這就是QLibrary內(nèi)部實現(xiàn)的代碼),可以看到
- bool load_sys();
- bool unload_sys();
- void *resolve_sys(const char *);
三個函數(shù)。在qlibrary.cpp中可以找到調(diào)用這三個函數(shù)的地方
- bool QLibrary::load ()調(diào)用了load_sys;
- bool QLibrary::unload ()調(diào)用了unload_sys;
- void * QLibrary::resolve ( const char * symbol )調(diào)用了resolve_sys
但是并沒有找到這三個函數(shù)的實現(xiàn),這是這么回事呢?
打開QTDIR\src\corelib\plugin\plugin.pri文件,
- win32 {
- SOURCES += plugin/qlibrary_win.cpp
- }
- unix {
- SOURCES += plugin/qlibrary_unix.cpp
- }
原來如此啊。
我們仔細看下qlibrary_win.cpp文件,load_sys函數(shù)調(diào)用了LoadLibrary,unload_sys調(diào)用了FreeLibrary,resolve_sys調(diào)用了GetProcAddress。
而在qlibrary_unix.cpp文件中,各種linux平臺又分好多種。但是基本上load_sys調(diào)用了dlopen,unload_sys調(diào)用了dlclose,resolve_sys調(diào)用了dlsym。
在HPUX中dlopen對應(yīng)shl_load,dlclose對應(yīng)shl_unload,dlsym對應(yīng)shl_findsym。
原來QLibrary就是這樣實現(xiàn)不同平臺動態(tài)庫的調(diào)用。
小結(jié):詳解 QT 源碼之 QLibrary 跨平臺調(diào)用動態(tài)庫實現(xiàn)的內(nèi)容介紹完了,希望本篇文章對你有所幫助!