库是共享程序代码的方式,主要有静态库(.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中,调用第三方库可以更好的实现代码复用,减少程序员的工作量。