库是共享程序代码的方式,主要有静态库(.lib),动态库(.dll)
静态库:链接时完整地拷贝至可执行文件中,被多次使用就有多份冗余拷贝。
动态库:链接时不复制,程序运行时由系统动态加载到内存,供程序调用,系统只加载一次,多个程序共用,节省内存。
QT调用DLL的方式:显式调用和隐式调用。没有导入库文件(,lib文件),只有头文件(.h)与动态链接库(.dll)时,显式调用,如果三个文件都全,使用隐式调用。
显式调用的三个函数:LoadLibrary()、GetProcAdress()、FreeLibrary();
LoadLibrary载入指定的dll文件,加载到调用程序的内存中。GetProcAdress函数指定动态链接库(dll)中的输出库函数地址,以备调用。FreeLibrary释放dll所占空间。
显式调用实例:
1 #include <QApplication>
2 #include <QLibrary>
3 #include <QDebug>
4 #include <QMessageBox>
5 #include "dll.h" //引入头文件
6 typedef int (*Fun)(int,int); //定义函数指针,以备调用
7 int main(int argc,char **argv)
8 {
9 QApplication app(argc,argv);
10 QLibrary mylib("myDLL.dll"); //声明所用到的dll文件
11 int result;
12 if (mylib.load()) //判断是否正确加载
13 {
14 QMessageBox::information(NULL,"OK","DLL load is OK!");
15 Fun open=(Fun)mylib.resolve("add"); //援引 add() 函数
16 if (open) //是否成功连接上 add() 函数
17 {
18 QMessageBox::information(NULL,"OK","Link to Function is OK!");
19 result=open(5,6); //这里函数指针调用dll中的 add() 函数
20 qDebug()<<result;
21 }
22 else
23 QMessageBox::information(NULL,"NO","Linke to Function is not OK!!!!");
24 }
25 else
26 QMessageBox::information(NULL,"NO","DLL is not loaded!");27 return 0; //加载失败则退出
28}
myDLL.dll为自定义的dll文件,将其复制到程序的输出目录下就可以调用。显然,显示调用代码书写量巨大,实在不方便。
隐式调用实例:
#include <QApplication>
2 #include <QDebug>
3 extern "C" //由于是C版的dll文件,在C++中引入其头文件要加extern "C" {},注意
4 {
5 #include "dll.h"
6 }
7 int main(int argv ,char **argv)
8 {
9 QApplication app(argv,argv);
10 HelloWordl(); //调用Win32 API 弹出helloworld对话框
11 qDebug()<<add(5,6); // dll 中我自己写的一个加法函数
12 return 0; //完成使命后,直接退出,不让它进入事件循环
13 }
相对于显式调用,隐式调用的代码书写更加简洁。但是隐式调用需要手动配置路径。
这个时候我们需要三个文件,头文件(.h)、导入库文件(.lib)、动态链接库(.dll),具体步骤如下:
1、首先我们把 .h 与 .lib/.a 文件复制到程序当前目录下,然后再把dll文件复制到程序的输出目录,
2、下面我们在pro文件中,添加 .lib 文件的位置:
LIBS+= -L D:/hitempt/api/ -l myDLL
-L 参数指定 .lib/.a 文件的位置
-l 参数指定导入库文件名(不要加扩展名)
另外,导入库文件的路径中,反斜杠用的是向右倾斜的
补充:
unix:LIBS += -L your_lib_path -lyour_lib(-l后面是库的名字去掉lib和后缀
比如libprotobuf.so,则为-lprotobuf)
win32:LIBS += your_lib_path/your_lib(mingw和msvc可能不一样)
3、在程序中include头文件(我试验用的dll是用C写的,因此要用 extern "C" { #include "dll.h" } )
补充:编译的时候没报错,说明 pro文件的参数没问题,可能是运行的时候找不到so,运行时大概的库搜索路径顺序
1)当前目录
2) LD_LIBRARY_PATH参数设定目录
3) /etc/ld.so 下配置文件设定,修改后需root用户调用下ldconfig刷新下
4)/lib ,/usr/lib目录
建议在qtcreator 里,project 里环境变量LD_LIBRARY_PATH增加 path_to_lib,或者直接把so文件拷贝到当前调用主程序当前目录
在qtcreator中,调用第三方库可以更好的实现代码复用,减少程序员的工作量。