gcc以及Qt的DLL文件调用之总结
2013-10-28 14:58
323 查看
DLL与LIB的区别 :
1.DLL是一个完整程序,其已经经过链接,即不存在同名引用,且有导出表,与导入表lib是一个代码集(也叫函数集)他没有链接,所以lib有冗余,当两个lib相链接时地址会重新建立,当然还有其它相关的不同,用lib.exe就知道了;
2.在生成dll时,经常会生成一个.lib(导入与导出),这个lib实际上不是真正的函数集,其每一个导出导入函数都是跳转指令,直接跳转到DLL中的位置,这个目的是外面的程序调用dll时自动跳转;
3.实际上最常用的lib是由lib.exe把*.obj生成的lib。(引用这里 )
Qt 想调用DLL怎么办呢?最先想到的是直接调用试试看:包含头文件,并把DLL文件拷到当前目录下,调用声明的函数,结果链接时和预想的一样出错了。修改 Makefile.debug,的链接选项也不行。Windows里的gcc调编译时用的库是和Linux一样的.a文件。其实,上面用VC直接调用 DLL的方法,在Windows里其实叫“隐式链接”,相对的叫“显式链接”,可以参考CSDN的解释 。
Qt调用DLL方法一:使用Win32
API
在显式链接下,应用程序必须进行函数调用以在运行时显式加载 DLL。为显式链接到 DLL,应用程序必须:
? 调用 LoadLibrary(或相似的函数)以加载 DLL 和获取模块句柄。
? 调用 GetProcAddress,以获取指向应用程序要调用的每个导出函数的函数指针。由于应用程序是通过指针调用 DLL 的函数,编译器不生成外部引用,故无需与导入库链接。
? 使用完 DLL 后调用 FreeLibrary。
例如:
typedef UINT (CALLBACK* LPFNDLLFUNC1)(DWORD,UINT);
HINSTANCE hDLL; // Handle to DLL
LPFNDLLFUNC1 lpfnDllFunc1; // Function pointer
DWORD dwParam1;
UINT uParam2, uReturnVal;
hDLL = LoadLibrary(”MyDLL”);
if (hDLL != NULL)
{
lpfnDllFunc1 = (LPFNDLLFUNC1)GetProcAddress(hDLL,
“DLLFunc1″);
if (!lpfnDllFunc1)
{
// handle the error
FreeLibrary(hDLL);
return SOME_ERROR_CODE;
}
else
{
// call the function
uReturnVal = lpfnDllFunc1(dwParam1, uParam2);
}
}
需要说明的是,typedef UINT (CALLBACK* LPFNDLLFUNC1)(DWORD,UINT);是声明想要调用的函数指针,这个函数的参数必须和DLL里面的一样。DLLFunc1是DLL里面 的“实际的函数名”,必须和头文件里面声明的一样,否则将会调用失败。使用需要加Windows.h 头文件。Windows里的Qt是用Mingw
GCC来编译,而Mingw GCC可以支持Win32的API。使用时并不需要包含头文件。这个方法没有实际验证,理论上是可以的。
Qt调用DLL方法二:使用Qt的API
对于调用DLL的方法,Qt原来本身就有相应的类来实现,用起来和Win32的步骤差不多。下面是代码,已经编译通过。在控制台依次输入qmake
–project、qmake、 nmake,即可。
#include
#include
#include
//动态链接不需要包含LTM8000D.h头文件
typedef int ( *pcom_open)(int , int , int ); //定义函数指针
int main(int argc, char *argv[])
{
int ret; //函数返回值
int port=1; //端口
int baud=0; //波特率
int rtsdtr=0; //串行口485方向控制设置:
QApplication a(argc, argv);
QLabel label1(”label-1″); //显示打印信息,将就着用了。。。
QLabel label2(”label-2″);
QLibrary mylib(”LTM8000D.DLL”);
if(mylib.load())
{
label1.setText(”load DLL success!…”);
pcom_open open=(pcom_open)mylib.resolve(”ltm_com_open”); //“ltm_com_open”为DLL包含的实际函数名,必须实际存在
if(open)
{
ret=open(port, baud, rtsdtr); //在这里调用DLL里的函数
label2.setText(”resolve ok…”);
}
else
{
label1.setText(”resolve failed…”);
}
}
else
{
label2.setText(”load DLL failed…”);
}
label1.show();
label2.show();
return a.exec();
}
Qt调用DLL方法三:直接调用 DLL
VC的引用库文件为xxx.lib, GCC的为xxx.a,通过比较两种库文件的格式,发现很相似。于是把xxx.dll,xxx.lib和xxx.h复制到Qt的project下,直接把xxx.lib改为xxx.a, 根据Qt的库名字的格式, 在xxx.a的前面加上lib, 即为libxxx.a。
再在Qt的.pro文件中最后面加上
LIBS += -L. –lxxx //增加当前目录下的libxxx.a
在Qt的源文件中加上
#include “xxx.h”
现在就可以直接调用xxx.h中的函数了。
// xxx.h
#ifndef XXX_H
#define XXX_H
WINAPI int xxx_func(void);
#endif
// main.cpp
#include “xxxh”
#include
#include
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QLabel label1(”label-1″); //显示打印信息,将就着用了。。。
if(xxx_func()) //“xxx_func”为DLL包含的实际函数名,必须实际存在
{
label1.setText(”Run DLL success!…”);
}
label1.show();
return a.exec();
}
总结: 类似于调用DLL的交叉编程使用的场合很多,还可以用Qt生成DLL,给其它工具调用。说到底,VC和Qt等本质上都是一样的,只是提供了不同的C++库。
1.DLL是一个完整程序,其已经经过链接,即不存在同名引用,且有导出表,与导入表lib是一个代码集(也叫函数集)他没有链接,所以lib有冗余,当两个lib相链接时地址会重新建立,当然还有其它相关的不同,用lib.exe就知道了;
2.在生成dll时,经常会生成一个.lib(导入与导出),这个lib实际上不是真正的函数集,其每一个导出导入函数都是跳转指令,直接跳转到DLL中的位置,这个目的是外面的程序调用dll时自动跳转;
3.实际上最常用的lib是由lib.exe把*.obj生成的lib。(引用这里 )
Qt 想调用DLL怎么办呢?最先想到的是直接调用试试看:包含头文件,并把DLL文件拷到当前目录下,调用声明的函数,结果链接时和预想的一样出错了。修改 Makefile.debug,的链接选项也不行。Windows里的gcc调编译时用的库是和Linux一样的.a文件。其实,上面用VC直接调用 DLL的方法,在Windows里其实叫“隐式链接”,相对的叫“显式链接”,可以参考CSDN的解释 。
Qt调用DLL方法一:使用Win32
API
在显式链接下,应用程序必须进行函数调用以在运行时显式加载 DLL。为显式链接到 DLL,应用程序必须:
? 调用 LoadLibrary(或相似的函数)以加载 DLL 和获取模块句柄。
? 调用 GetProcAddress,以获取指向应用程序要调用的每个导出函数的函数指针。由于应用程序是通过指针调用 DLL 的函数,编译器不生成外部引用,故无需与导入库链接。
? 使用完 DLL 后调用 FreeLibrary。
例如:
typedef UINT (CALLBACK* LPFNDLLFUNC1)(DWORD,UINT);
HINSTANCE hDLL; // Handle to DLL
LPFNDLLFUNC1 lpfnDllFunc1; // Function pointer
DWORD dwParam1;
UINT uParam2, uReturnVal;
hDLL = LoadLibrary(”MyDLL”);
if (hDLL != NULL)
{
lpfnDllFunc1 = (LPFNDLLFUNC1)GetProcAddress(hDLL,
“DLLFunc1″);
if (!lpfnDllFunc1)
{
// handle the error
FreeLibrary(hDLL);
return SOME_ERROR_CODE;
}
else
{
// call the function
uReturnVal = lpfnDllFunc1(dwParam1, uParam2);
}
}
需要说明的是,typedef UINT (CALLBACK* LPFNDLLFUNC1)(DWORD,UINT);是声明想要调用的函数指针,这个函数的参数必须和DLL里面的一样。DLLFunc1是DLL里面 的“实际的函数名”,必须和头文件里面声明的一样,否则将会调用失败。使用需要加Windows.h 头文件。Windows里的Qt是用Mingw
GCC来编译,而Mingw GCC可以支持Win32的API。使用时并不需要包含头文件。这个方法没有实际验证,理论上是可以的。
Qt调用DLL方法二:使用Qt的API
对于调用DLL的方法,Qt原来本身就有相应的类来实现,用起来和Win32的步骤差不多。下面是代码,已经编译通过。在控制台依次输入qmake
–project、qmake、 nmake,即可。
#include
#include
#include
//动态链接不需要包含LTM8000D.h头文件
typedef int ( *pcom_open)(int , int , int ); //定义函数指针
int main(int argc, char *argv[])
{
int ret; //函数返回值
int port=1; //端口
int baud=0; //波特率
int rtsdtr=0; //串行口485方向控制设置:
QApplication a(argc, argv);
QLabel label1(”label-1″); //显示打印信息,将就着用了。。。
QLabel label2(”label-2″);
QLibrary mylib(”LTM8000D.DLL”);
if(mylib.load())
{
label1.setText(”load DLL success!…”);
pcom_open open=(pcom_open)mylib.resolve(”ltm_com_open”); //“ltm_com_open”为DLL包含的实际函数名,必须实际存在
if(open)
{
ret=open(port, baud, rtsdtr); //在这里调用DLL里的函数
label2.setText(”resolve ok…”);
}
else
{
label1.setText(”resolve failed…”);
}
}
else
{
label2.setText(”load DLL failed…”);
}
label1.show();
label2.show();
return a.exec();
}
Qt调用DLL方法三:直接调用 DLL
VC的引用库文件为xxx.lib, GCC的为xxx.a,通过比较两种库文件的格式,发现很相似。于是把xxx.dll,xxx.lib和xxx.h复制到Qt的project下,直接把xxx.lib改为xxx.a, 根据Qt的库名字的格式, 在xxx.a的前面加上lib, 即为libxxx.a。
再在Qt的.pro文件中最后面加上
LIBS += -L. –lxxx //增加当前目录下的libxxx.a
在Qt的源文件中加上
#include “xxx.h”
现在就可以直接调用xxx.h中的函数了。
// xxx.h
#ifndef XXX_H
#define XXX_H
WINAPI int xxx_func(void);
#endif
// main.cpp
#include “xxxh”
#include
#include
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QLabel label1(”label-1″); //显示打印信息,将就着用了。。。
if(xxx_func()) //“xxx_func”为DLL包含的实际函数名,必须实际存在
{
label1.setText(”Run DLL success!…”);
}
label1.show();
return a.exec();
}
总结: 类似于调用DLL的交叉编程使用的场合很多,还可以用Qt生成DLL,给其它工具调用。说到底,VC和Qt等本质上都是一样的,只是提供了不同的C++库。
相关文章推荐
- Windows下gcc以及Qt的DLL文件调用之总结
- Windows下gcc以及Qt的DLL文件调用之总结
- Windows下gcc以及Qt的DLL文件调用之总结
- Windows下gcc以及Qt的DLL文件调用之总结
- qt中调用dll文件的方法
- vs下opencv的人脸检测以及qt下dll的封装调用(二、dll的创建)
- unity中调用dll文件总结
- QSS总结以及最近做的Qt项目
- 关于gcc、glibc和binutils模块之间的关系,以及在现有系统上如何升级的总结
- JNI学习总结之:java调用native函数以及native函数回调java类成员方法
- 关于gcc、glibc和binutils模块之间的关系,以及在现有系统上如何升级的总结 .
- Qt DLL总结-创建及调用QT的 DLL
- Qt调用QGIS出现的问题总结
- Windows下获取Dump文件以及进程下各线程调用栈的方法总结(转)
- vs2008, qt生成库文件 以及qt应用程序调用库文件方法
- QT 总结(四) 1.中文编码 乱码 2.QFileDialog 打开文件框 3.调用其它UI文件 4.QMessageBox弹窗
- 快排总结 原代码 以及 如何调用
- 关于gcc、glibc和binutils模块之间的关系,以及在现有系统上如何升级的总结
- Qt刷新机制的一些总结(Qt内部画的时候是相当于画在后台一个对象里,然后在刷新的时候调用bitblt统一画,调用window的api并不会影响到后面的那个对象)
- objective-c类以及方法调用的基本总结