加载dll、lib库(例子的代码很全)

是关于如何加载dll或lib库的。可以看这篇bog   Qt调用dll中的功能函数点击打开链接

****************************************************************************************************************************************************

声明: 事先我已经自己动手写了一个简单的dll文件(myDLL.dll),C版接口的。并且用我前两篇有关DLL文章里面的方法,从dll中导出了导入库(.lib)文件,dll中有两个函数,原型如下:

   void HelloWorld();     //函数内部调用Win32 API,功能是弹出一个helloworld提示框   int add(int a,int b);  //实现两个数相加,并返回结果

 下面分别通过显示调用和隐式调用两种方法,来模拟Qt如何调用外部dll文件中的功能函数,follow me....

预备知识:

     1、如果在没有导入库文件(.lib),而只有头文件(.h)与动态链接库(.dll)时,我们才需要显示调用,如果这三个文件都全的话,我们就可以使用简单方便的隐式调用。

     2、通常Windows下程序显示调用dll的步骤分为三步(三个函数):LoadLibrary()、GetProcAdress()、FreeLibrary()

 其中,LoadLibrary() 函数用来载入指定的dll文件,加载到调用程序的内存中(DLL没有自己的内存!)

         GetProcAddress() 函数检索指定的动态链接库(DLL)中的输出库函数地址,以备调用

         FreeLibrary() 释放dll所占空间 

1、显示调用 

      Qt提供了一个 QLibrary 类供显示调用。下面给出一个完整的例子:

[cpp] view plain copy
 
  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文件,将其复制到程序的输出目录下就可以调用。显然,显示调用代码书写量巨大,实在不方便。

2、隐式调用

    这个时候我们需要三个文件,头文件(.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" } )

 下面是隐式调用的实例代码:

[cpp] view plain copy
 
  1. #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. }  


************************************************************************************************************************************

本来是打算用隐身调用的,但是可能是库的原因,造成无法使用。库是别的公司提供的。so,用的第一种。在后面会贴一些我的code。

jida.h 部分 函数形成如下

[cpp] view plain copy
 
  1. BOOL WINAPI JidaDllInstall(BOOL install);  
  2. BOOL WINAPI JidaDllInitialize(void);  
  3. BOOL WINAPI JidaDllUninitialize(void);  
  4.   
  5. DWORD WINAPI JidaBoardCountW(LPCWSTR pszClass, DWORD dwFlags);  
  6. BOOL WINAPI JidaBoardOpenW(LPCWSTR pszClass, DWORD dwNum, DWORD dwFlags,/*@out@*/PHJIDA phJida);  
  7. BOOL WINAPI JidaBoardClose(HJIDA hJida);  
  8.   
  9. DWORD WINAPI JidaI2CCount(HJIDA hJida);  
  10. DWORD WINAPI JidaI2CType(HJIDA hJida, DWORD dwType);  
  11. BOOL WINAPI JidaI2CRead(HJIDA hJida, DWORD dwType, BYTE bAddr, LPBYTE pBytes,DWORD dwLen);  
  12. BOOL WINAPI JidaI2CWrite(HJIDA hJida, DWORD dwType, BYTE bAddr, LPBYTE pBytes,DWORD dwLen);  

在要加载的cpp文件上写上QT,因为dll是window的,里面含有定义的宏DWORD这类,so加上windows.h

[cpp] view plain copy
 
  1. #include <windows.h>  
  2. #include <QLibrary>  
  3. #include "jida.h"  


定义指针函数以备调用如:一个add函数。下面第一行为函数,第二行为函数指针。

[cpp] view plain copy
 
  1. //int add(int a,int b);  
  2. //typedef int (*Fun)(int,int); //定义函数指针,以备调用  


下面是我的main.cpp里面的。第一行pHandle是给dll的函数里面的api接口,不同dll接口不同,请忽略,不用在意。

主要是dll的加载。QLibrary jida_lib("jida.dll");

然后引用dll里面的函数。resolve是加载dll里面的函数。括号里面的(JidaDallInstall)是Dell里面的函数名。然后直接调用jida_install即可使用dll里面的JidaDallInstall函数。jida_install是自己自定义的。

[cpp] view plain copy
 
  1. //加载dll 函数  
  2.     FunJidaInstal jida_install=(FunJidaInstal)jida_lib.resolve("JidaDllInstall");    //引用dll里面的JidaDllInstall() 函数  

DWORD pHandle = 0; //句柄返回值

[cpp] view plain copy
 
  1. //int add(int a,int b);  
  2. //typedef int (*Fun)(int,int); //定义函数指针,以备调用  
  3.   
  4. //dll init uninit  
  5. typedef BOOL WINAPI (*FunJidaInstal)(BOOL install);     //BOOL WINAPI JidaDllInstall(BOOL install);  
  6. typedef BOOL WINAPI (*FunJidaDllInit)(BOOL install);    //BOOL WINAPI JidaDllInitialize(void);  
  7. typedef BOOL WINAPI (*FunJidaDllUninit)(BOOL install);  //BOOL WINAPI JidaDllUninitialize(void);  
  8. //板子  
  9. typedef DWORD WINAPI (*FunJidaBoardCountW)(LPCWSTR,DWORD);             //DWORD WINAPI JidaBoardCountW(LPCWSTR pszClass, DWORD dwFlags); 板子数  
  10. typedef BOOL WINAPI (*FunJidaBoardOpenW)(LPCWSTR,DWORD,DWORD,PHJIDA);//打开板子 BOOL WINAPI JidaBoardOpenW(LPCWSTR pszClass, DWORD dwNum, DWORD dwFlags,/*@out@*/PHJIDA phJida);  
  11. typedef BOOL WINAPI (*FunJidaBoardClose)(HJIDA hJida);//BOOL WINAPI JidaBoardClose(HJIDA hJida); 板子关闭  
  12. //I2C  
  13. typedef DWORD WINAPI (*FunJidaI2CCount)(HJIDA);              //DWORD WINAPI JidaI2CCount(HJIDA hJida);  
  14. typedef DWORD WINAPI (*FunJidaI2CType)(HJIDA, DWORD ); //DWORD WINAPI JidaI2CType(HJIDA hJida, DWORD dwType);  
  15. typedef BOOL WINAPI (*FunJidaI2CRead)(HJIDA, DWORD , BYTE , LPBYTE ,DWORD );//BOOL WINAPI JidaI2CRead(HJIDA hJida, DWORD dwType, BYTE bAddr, LPBYTE pBytes,DWORD dwLen);  
  16. typedef BOOL WINAPI (*FunJidaI2CWrite)(HJIDA, DWORD , BYTE , LPBYTE ,DWORD );//BOOL WINAPI JidaI2CWrite(HJIDA hJida, DWORD dwType, BYTE bAddr, LPBYTE pBytes,DWORD dwLen);  
  17.   
  18. //声明所用到的dll文件  
  19. QLibrary jida_lib("jida.dll");  
  20. //加载dll 函数  
  21. FunJidaInstal jida_install=(FunJidaInstal)jida_lib.resolve("JidaDllInstall");    //引用dll里面的JidaDllInstall() 函数  
  22. FunJidaDllInit jida_dll_init = (FunJidaDllInit)jida_lib.resolve("JidaDllInitialize");  
  23. FunJidaDllUninit jida_dll_uninit = (FunJidaDllUninit)jida_lib.resolve("JidaDllUninitialize");  
  24. FunJidaBoardCountW jida_board_countw = (FunJidaBoardCountW)jida_lib.resolve("JidaBoardCountW");  
  25. FunJidaBoardOpenW jida_board_openw = (FunJidaBoardOpenW)jida_lib.resolve("JidaBoardOpenW");  
  26. FunJidaBoardClose jida_board_close = (FunJidaBoardClose)jida_lib.resolve("JidaBoardClose");  
  27. FunJidaI2CCount jida_i2c_count = (FunJidaI2CCount)jida_lib.resolve("JidaI2CCount");  
  28. FunJidaI2CType jida_i2c_type = (FunJidaI2CType)jida_lib.resolve("JidaI2CType");  
  29. FunJidaI2CRead jida_i2c_read = (FunJidaI2CRead)jida_lib.resolve("JidaI2CRead");  
  30. FunJidaI2CWrite jida_i2c_write = (FunJidaI2CWrite)jida_lib.resolve("JidaI2CWrite");  



进行下简单的判断,dll是否加载成功。.load是加载dll。jida_install是我调用dll里面的dll初始化程序。

等于是调用dll里面的函数,再次看是否加载ok。

[cpp] view plain copy
 
    1. if (jida_lib.load()) //判断是否正确加载  
    2. {  
    3.         cout << "qt lib load ok" << endl;  
    4. }  
    5. else  
    6. {  
    7.     cout << "qt lib load error" << endl;  
    8. }  
    9.   
    10. if(jida_install)  
    11. {  
    12.     cout << "dell load ok" <<endl;  
    13. }  
    14. else  
    15. {  
    16.     cout << "dell load error" <<endl;  
    17. }  

http://blog.csdn.net/linbounconstraint/article/details/47321103

原文地址:https://www.cnblogs.com/findumars/p/7011187.html