Qt添加鏈接庫總結
一 前言:
在Qt工程開發過程中,經常會碰到集成第三方庫的情況,不論是靜態庫還是動態庫。由於Qt本身不完善等特性,集成過程中經常會出現各種各樣的問題。針對遇到過的問題記錄如下備忘,同時也希望能為更多開發者朋友提供前車之鑒。
二 基礎知識簡介:
2.1關於lib和dll的區別如下:
(1)lib是編譯時用到的,dll是運行時用到的。如果要完成源代碼的編譯,只需要 lib;如果要使動態鏈接的程序運行起來,只需要dll。
(2)如果有dll文件,那么lib一般是一些索引信息,記錄了dll中函數的入口和位 置,dll中是函數的具體內容;如果只有lib文件,那么這個lib文件是靜態編譯出來的,索引和實現都在其中。使用靜態編譯的lib文件,在運行程序時 不需要再掛動態庫,缺點是導致應用程序比較大,而且失去了動態庫的靈活性,發布新版本時要發布新的應用程序才行。
(3)動態鏈接的情況下,有兩個 文件:一個是LIB文件,一個是DLL文件。LIB包含被DLL導出的函數名稱和位置,DLL包含實際的函數和數據,應用程序使用LIB文件鏈接到DLL 文件。在應用程序的可執行文件中,存放的不是被調用的函數代碼,而是DLL中相應函數代碼的地址,從而節省了內存資源。DLL和LIB文件必須隨應用程序 一起發行,否則應用程序會產生錯誤。如果不想用lib文件或者沒有lib文件,可以用WIN32 API函數LoadLibrary、GetProcAddress裝載。
三 加載方式:
3.1鏈接的兩種方式
轉:
顯示調用
(該方式只使適用只有dll文件的例子,在此情況下,dll實現的函數名及參數應該是已知的)
Qt提供了一個 QLibrary 類供顯示調用。下面給出一個完整的例子:
#include #include #include #include #include “dll.h” //引入頭文件 typedef int (*Fun)(int,int); //定義函數指針,以備調用 int main(int argc,char **argv) { QApplication app(argc,argv); QLibrary mylib(“myDLL.dll”); //聲明所用到的dll文件 int result; if (mylib.load()) //判斷是否正確加載 { QMessageBox::information(NULL,“OK”,“DLL load is OK!”); Fun open=(Fun)mylib.resolve(“add”); //援引 add() 函數 if (open) //是否成功連接上 add() 函數 { QMessageBox::information(NULL,“OK”,“Link to Function is OK!”); result=open(5,6); //這里函數指針調用dll中的 add() 函數 qDebug()<<result; } else QMessageBox::information(NULL,“NO”,“Linke to Function is not OK!!!”); } else QMessageBox::information(NULL,“NO”,“DLL is not loaded!”); return 0; //加載失敗則退出
myDLL.dll為自定義的dll文件,將其復制到程序的輸出目錄下就可以調用。顯然,顯示調用代碼書寫量巨大,實在不方便。
隱式調用
這個時候我們需要三個文件,頭文件(.h)、導入庫文件(.lib)、動態鏈接庫(.dll),具體步驟如下:
1、首先我們把 .h 與 .lib/.a 文件復制到程序當前目錄下,然后再把dll文件復制到程序的輸出目錄,
2、下面我們在pro文件中,添加 .lib 文件的位置: LIBS+= -L D:/hitempt/api/ -l myDLL
-L 參數指定 .lib/.a 文件的位置
-l 參數指定導入庫文件名(不要加擴展名)
另外,導入庫文件的路徑中,反斜杠用的是向右傾斜的
3、在程序中include頭文件(我試驗用的dll是用C寫的,因此要用 extern “C” { #include “dll.h” } )
下面是隱式調用的實例代碼:
#include #include extern “C” //由於是C版的dll文件,在C++中引入其頭文件要加extern “C” {},注意 { #include “dll.h” } int main(int argv ,char **argv) { QApplication app(argv,argv); HelloWordl(); //調用Win32 API 彈出helloworld對話框 qDebug()<<add(5,6); // dll 中我自己寫的一個加法函數 return 0; //完成使命后,直接退出,不讓它進入事件循環 }
3.2lib地址在pro文件中的寫法
網友提供了一下這種方法,其中相當有借鑒之處。但我用后發現卻編譯失敗(Qt5.6.1 Qt creator4.5.1),主要原因在與:lib文件的地址必須使用相對地址,而不是絕對地址,將一下地址改為:
LIBS += -L…/與工程目錄同級的、放置lib的文件夾名/ -l后面緊跟lib文件名
就通過了。
在.pro文件中,加上一句話,告訴工程,.lib在哪里
LIBS += -LE:/project/QT/usbcan_info/ -lControlCAN
下面對這句話進行解析:
1)+=這個符號是連接在一起的,不要自做聰明,給分開了,寫成+ = ,這樣就是錯的!
2)+=的左右兩邊可以有空格,也可以沒有空格,隨意的
3)-L之后緊接着就跟着.lib文件所在的目錄,比如,我的目錄是在E盤下的project目錄下的QT目錄下的usbcan_info下,注意,是緊跟着,不要分開,分開就錯啦!
4)-l后面跟着的是.lib的文件名,注意,比如你的.lib文件時ControlCAN.lib,不要傻乎乎的寫成了-lControlCAN.lib,不要帶后綴,要寫成-lControlCAN,同樣,-l后面也沒有空格
3.3關於相對目錄的問題
在上一點中提到的相對目錄的表示,例如-L …/debug/ ,debug不是當前工程所處的文件夾下的debug文件,而是和工程所在目錄同級的,因此一定要記清楚。