最近这阵子,接了个私活,封装一个开发包俗称的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
运行结果:
CSDN下载地址:http://download.csdn.net/detail/feiyangqingyun/7327095