标签:
最近这阵子,接了个私活,封装一个开发包俗称的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代码如下
#include "qmfcapp.h" #include "windows.h" //windows下DLL函数入口 BOOL WINAPI DllMain( HINSTANCE hInstance, DWORD dwReason, LPVOID lpvReserved ) { static bool ownApplication = FALSE; if ( dwReason == DLL_PROCESS_ATTACH ) ownApplication = QMfcApp::pluginInstance( hInstance ); if ( dwReason == DLL_PROCESS_DETACH && ownApplication ) delete qApp; return TRUE; } //定义函数回调指针 typedef void (CALLBACK *FunCallBack)(int par); //定义回调函数 FunCallBack OnEvent=NULL; //定义回调参数 int par; //定义C语言类型导出事件函数 extern "C" __declspec(dllexport) void SetFunCallBack(FunCallBack fun,int p) { OnEvent=fun; par=p; } //定义C语言类型导出函数 extern "C" __declspec(dllexport) int add(int i) { //下面这段代码用来触发事件,如果仅仅是使用add函数的话可以删除. if (i==110){ if (OnEvent){ //当传入参数为110而且已经定义过回调函数的话,则触发事件. OnEvent(120); } } return i*2; }View Code
选择release编译,在目录下生成了一个DLL文件,这个文件就可以供VC/C#/VB/JAVA等调用了。
第四步:编写VC和C语言程序测试
main.c代码如下:
#include <stdio.h> #include <windows.h> //定义函数指针 typedef int (*Add)(int); //定义回调函数指针 typedef void (CALLBACK *FunCallBack)(int); //定义回调函数处理方法 void CALLBACK HandleEvent(int par) { printf("%d\n", par); } //主函数入口 int main(int argc, char const *argv[]) { //定义DLL句柄 HINSTANCE hDll = LoadLibrary("qtdll.dll"); if (hDll != NULL) { //定义回调函数指针 typedef void (CALLBACK *PFunCallBack)(FunCallBack); //实例化回调函数指针 PFunCallBack SetFunCallBack=(PFunCallBack)GetProcAddress(hDll,"SetFunCallBack"); //执行回调函数 if (SetFunCallBack){ SetFunCallBack(HandleEvent); } //实例化函数指针 Add add = (Add)GetProcAddress(hDll,"add"); //调用DLL中的方法 printf("%d\n", add(1)); //调用DLL中的方法并触发事件 printf("%d\n", add(110)); FreeLibrary(hDll); } return 0; }View Code
运行结果如图:
当传入的参数为110时,会触发事件,返回值120。
第五步:编写C#程序测试
using System; using System.Collections.Generic; using System.Text; using System.Runtime.InteropServices; namespace qtdllTestNET { class Program { //声明DLL中的函数 [DllImport("qtdll.dll")] public static extern int add(int i); //声明DLL中的回调函数,即事件 [DllImport("qtdll.dll")] public static extern void SetFunCallBack([MarshalAs(UnmanagedType.FunctionPtr)] CallbackFun pCallbackFun); [UnmanagedFunctionPointer(CallingConvention.Cdecl)] public delegate void CallbackFun(int i); public static void HandleEvent(int i) { Console.WriteLine(i); } static void Main(string[] args) { CallbackFun HandleEventX = HandleEvent; SetFunCallBack(HandleEventX); Console.WriteLine(add(1)); Console.WriteLine(add(110)); Console.ReadLine(); } } }View Code
运行结果:
CSDN下载地址:http://download.csdn.net/detail/feiyangqingyun/7327095
http://www.cnblogs.com/feiyangqingyun/p/3720777.html
QT编写DLL给外部程序调用,提供VC/C#/C调用示例(含事件)
标签:
原文地址:http://www.cnblogs.com/findumars/p/5055355.html