QT编写DLL给外部程序调用,提供VC/C#/C调用示例(含事件)
最近这阵子,接了个私活,封装一个开发包俗称的SDK给客户调用,查阅了很多人家的SDK,绝大部分用VC编写,而且VC6.0居多,估计也是为了兼容大量的XP用户及IE浏览器,XP自带了VC6.0运行库,所以无需兼带其他DLL即可运行。也发现有些用汇编和C封装的,那估计是高手才能编写啊!那怎么办呢?眼看一星期的交货期要到了,由于本人从来没有写过SDK开发包,心里那个着急啊,赶紧想出了三种处理方案。
方案一:
用自己最熟悉的QT来写,毕竟QT也是C++的一种嘛,应该可以和VC通用的。初步实验,你妹啊,调用不起来,直接废弃。
方案二:
找VC高手,自己掏钱出来找人写,客户是上帝,那可不能耽误呢!各大群里发消息,无奈高手要价太高(一个简单的socket通信封装,协议都有,熟手估计半天,要价8K有点高),绝大部分答话者使用.NET和JAVA语言,这个编写DLL肯定不大适合,你想想提供个DLL还要客户安装个框架,情何以堪!话说众里寻他千百度啊,我一个亲戚就是专门搞VC开发12年啦,这点程序半天就可以搞定的,找到之后,时间有限太忙了,没时间写,郁闷!再次放弃!~
方案三:
时间一分一秒过去,真心着急啊,难不成自己用VC编写?有想法就赶紧行动,找到了古董级别的VS6.0的安装光盘,双击看到阔别七年的安装界面,突然之间有种热泪盈眶的感觉。老朋友啊,当年的编程入门,又记起了当年的大学,当年的图书馆,当年的初恋!
搞就搞,新建MFC DLL,我的妈呀,VC的命名规则实在让人抓狂!半自动语法提示更是让我有种自杀的冲动!安装了visualAssistX,还是不爽!话说编程也要带着快乐的心情嘛,所以果断放弃!
怎么办呢?既然QT也是C++,而且有个大名鼎鼎的WPS都是QT和MFC混合编程的,所以肯定有方法的,百度谷歌,找到一个qtwinmigrate的东西,在这里万分感谢qtwinmigrate的作者!
好吧,参考里面的例子,正式开始编写了,咱就先来一个最初级的吧。
第一步:
打开qt creator,新建C++ 库项目
依次下一步,记得在选择需要的模块的时候选择QtGui模块,如果没有选中的话,后面编译通不过,qmfcapp这个类里面用了QtGui模块中的方法。
第二步:
将多余的文件删除,_global.h还有其他两个文件删除,因为这是QT新建DLL自带的,我们要写的是可以供其他语言调用的DLL,我这里的例子是只用了一个函数和事件,没有用到界面的东西,所以从qtwinmigrate下面的src文件夹拷贝qmfcapp.h和qmfcapp.cpp这两个文件过来即可,如下。
第三步:
更改pro文件,改为
TARGET = qtdll
TEMPLATE = lib
CONFIG += dll
SOURCES += \
qmfcapp.cpp \
main.cpp
HEADERS +=\
qmfcapp.h
main.cpp代码如下
1 #include "qmfcapp.h" 2 #include "windows.h" 3 4 //windows下DLL函数入口 5 BOOL WINAPI DllMain( HINSTANCE hInstance, DWORD dwReason, LPVOID lpvReserved ) 6 { 7 static bool ownApplication = FALSE; 8 9 if ( dwReason == DLL_PROCESS_ATTACH ) 10 ownApplication = QMfcApp::pluginInstance( hInstance ); 11 if ( dwReason == DLL_PROCESS_DETACH && ownApplication ) 12 delete qApp; 13 14 return TRUE; 15 } 16 17 //定义函数回调指针 18 typedef void (CALLBACK *FunCallBack)(int par); 19 //定义回调函数 20 FunCallBack OnEvent=NULL; 21 //定义回调参数 22 int par; 23 24 //定义C语言类型导出事件函数 25 extern "C" __declspec(dllexport) void SetFunCallBack(FunCallBack fun,int p) 26 { 27 OnEvent=fun; 28 par=p; 29 } 30 31 //定义C语言类型导出函数 32 extern "C" __declspec(dllexport) int add(int i) 33 { 34 //下面这段代码用来触发事件,如果仅仅是使用add函数的话可以删除. 35 if (i==110){ 36 if (OnEvent){ 37 //当传入参数为110而且已经定义过回调函数的话,则触发事件. 38 OnEvent(120); 39 } 40 } 41 42 return i*2; 43 }
View Code
选择release编译,在目录下生成了一个DLL文件,这个文件就可以供VC/C#/VB/JAVA等调用了。
第四步:编写VC和C语言程序测试
main.c代码如下:
1 #include <stdio.h> 2 #include <windows.h> 3 4 //定义函数指针 5 typedef int (*Add)(int); 6 //定义回调函数指针 7 typedef void (CALLBACK *FunCallBack)(int); 8 9 //定义回调函数处理方法 10 void CALLBACK HandleEvent(int par) 11 { 12 printf("%d\n", par); 13 } 14 15 //主函数入口 16 int main(int argc, char const *argv[]) 17 { 18 //定义DLL句柄 19 HINSTANCE hDll = LoadLibrary("qtdll.dll"); 20 if (hDll != NULL) 21 { 22 //定义回调函数指针 23 typedef void (CALLBACK *PFunCallBack)(FunCallBack); 24 25 //实例化回调函数指针 26 PFunCallBack SetFunCallBack=(PFunCallBack)GetProcAddress(hDll,"SetFunCallBack"); 27 //执行回调函数 28 if (SetFunCallBack){ 29 SetFunCallBack(HandleEvent); 30 } 31 32 //实例化函数指针 33 Add add = (Add)GetProcAddress(hDll,"add"); 34 //调用DLL中的方法 35 printf("%d\n", add(1)); 36 37 //调用DLL中的方法并触发事件 38 printf("%d\n", add(110)); 39 40 FreeLibrary(hDll); 41 } 42 return 0; 43 }
View Code
运行结果如图:
当传入的参数为110时,会触发事件,返回值120。
第五步:编写C#程序测试
1 using System; 2 using System.Collections.Generic; 3 using System.Text; 4 using System.Runtime.InteropServices; 5 6 namespace qtdllTestNET 7 { 8 class Program 9 { 10 //声明DLL中的函数 11 [DllImport("qtdll.dll")] 12 public static extern int add(int i); 13 //声明DLL中的回调函数,即事件 14 [DllImport("qtdll.dll")] 15 public static extern void SetFunCallBack([MarshalAs(UnmanagedType.FunctionPtr)] CallbackFun pCallbackFun); 16 [UnmanagedFunctionPointer(CallingConvention.Cdecl)] 17 public delegate void CallbackFun(int i); 18 19 public static void HandleEvent(int i) 20 { 21 Console.WriteLine(i); 22 } 23 24 static void Main(string[] args) 25 { 26 CallbackFun HandleEventX = HandleEvent; 27 SetFunCallBack(HandleEventX); 28 Console.WriteLine(add(1)); 29 Console.WriteLine(add(110)); 30 Console.ReadLine(); 31 } 32 } 33 }
View Code
运行结果: