• 21658阅读
  • 28回复

QT编写DLL给外部程序调用,提供VC/C#/C调用示例(含事件) [复制链接]

上一主题 下一主题
离线liudianwu
 

图酷模式  只看楼主 倒序阅读 楼主  发表于: 2014-05-10
— 本帖被 XChinux 执行加亮操作(2016-09-18) —
最近这阵子,接了个私活,封装一个开发包俗称的SDK给客户调用,查阅了很多人家的SDK,绝大部分用VC编写,而且VC6.0居多,估计也是为了兼容大量的XP用户及IE浏览器,XP自带了VC6.0运行库,所以无需兼带其他DLL即可运行。也发现有些用汇编和C封装的,那估计是高手才能编写啊!那怎么办呢?眼看一星期的交货期要到了,由于本人从来没有写过SDK开发包,心里那个着急啊,赶紧想出了三种处理方案。

方案一:
用自己最熟悉的QT来写,毕竟QT也是C++的一种嘛,应该可以和VC通用的。初步实验,你妹啊,调用不起来,直接废弃。

方案二:
VC高手,自己掏钱出来找人写,客户是上帝,那可不能耽误呢!各大群里发消息,无奈高手要价太高(一个简单的socket通信封装,协议都有,熟手估计半天,要价8K有点高),绝大部分答话者使用.NETJAVA语言,这个编写DLL肯定不大适合,你想想提供个DLL还要客户安装个框架,情何以堪!话说众里寻他千百度啊,我一个亲戚就是专门搞VC开发12年啦,这点程序半天就可以搞定的,找到之后,时间有限太忙了,没时间写,郁闷!再次放弃!~

方案三:
时间一分一秒过去,真心着急啊,难不成自己用VC编写?有想法就赶紧行动,找到了古董级别的VS6.0的安装光盘,双击看到阔别七年的安装界面,突然之间有种热泪盈眶的感觉。老朋友啊,当年的编程入门,又记起了当年的大学,当年的图书馆,当年的初恋!
window.open('http://www.qtcn.org/bbs/attachment/Mon_1405/44_110085_2134fc1bfd2e3f9.png?52');" style="max-width:700px;max-height:700px;" onload="if(is_ie6&&this.offsetWidth>700)this.width=700;" >

搞就搞,新建MFC DLL,我的妈呀,VC的命名规则实在让人抓狂!半自动语法提示更是让我有种自杀的冲动!安装了visualAssistX,还是不爽!话说编程也要带着快乐的心情嘛,所以果断放弃!

怎么办呢?既然QT也是C++,而且有个大名鼎鼎的WPS都是QTMFC混合编程的,所以肯定有方法的,百度谷歌,找到一个qtwinmigrate的东西,在这里万分感谢qtwinmigrate的作者!
好吧,参考里面的例子,正式开始编写了,咱就先来一个最初级的吧。

第一步:
打开qt creator,新建C++ 库项目


依次下一步,记得在选择需要的模块的时候选择QtGui模块,如果没有选中的话,后面编译通不过,qmfcapp这个类里面用了QtGui模块中的方法。

第二步:
将多余的文件删除,_global.h还有其他两个文件删除,因为这是QT新建DLL自带的,我们要写的是可以供其他语言调用的DLL,我这里的例子是只用了一个函数和事件,没有用到界面的东西,所以从qtwinmigrate下面的src文件夹拷贝qmfcapp.hqmfcapp.cpp这两个文件过来即可,如下。



第三步:
更改pro文件,改为
  1. TARGET = qtdll
  2. TEMPLATE = lib
  3. CONFIG += dll
  4. SOURCES += \
  5.     qmfcapp.cpp \
  6.     main.cpp
  7. HEADERS +=\
  8. qmfcapp.h




main.cpp代码如下
  1. #include "qmfcapp.h"
  2. #include "windows.h"
  3. //windows下DLL函数入口
  4. BOOL WINAPI DllMain( HINSTANCE hInstance, DWORD dwReason, LPVOID lpvReserved )
  5. {
  6.     static bool ownApplication = FALSE;
  7.     if ( dwReason == DLL_PROCESS_ATTACH )
  8.         ownApplication = QMfcApp::pluginInstance( hInstance );
  9.     if ( dwReason == DLL_PROCESS_DETACH && ownApplication )
  10.         delete qApp;
  11.     return TRUE;
  12. }
  13. //定义函数回调指针
  14. typedef void (CALLBACK *FunCallBack)(int par);
  15. //定义回调函数
  16. FunCallBack OnEvent=NULL;
  17. //定义回调参数
  18. int par;
  19. //定义C语言类型导出事件函数
  20. extern "C" __declspec(dllexport) void SetFunCallBack(FunCallBack fun,int p)
  21. {
  22.     OnEvent=fun;
  23.     par=p;
  24. }
  25. //定义C语言类型导出函数
  26. extern "C" __declspec(dllexport) int add(int i)
  27. {
  28.     //下面这段代码用来触发事件,如果仅仅是使用add函数的话可以删除.
  29.     if (i==110){
  30.         if (OnEvent){
  31.             //当传入参数为110而且已经定义过回调函数的话,则触发事件.
  32.             OnEvent(120);
  33.         }
  34.     }
  35.     return i*2;
  36. }




选择release编译,在目录下生成了一个DLL文件,这个文件就可以供VC/C#/VB/JAVA等调用了。

第四步:编写VCC语言程序测试
main.c代码如下:
  1. #include <stdio.h>
  2. #include <windows.h>
  3. //定义函数指针
  4. typedef int (*Add)(int);
  5. //定义回调函数指针
  6. typedef void (CALLBACK *FunCallBack)(int);
  7. //定义回调函数处理方法
  8. void CALLBACK HandleEvent(int par)
  9. {
  10. printf("%d\n", par);
  11. }
  12. //主函数入口
  13. int main(int argc, char const *argv[])
  14. {
  15. //定义DLL句柄
  16. HINSTANCE hDll = LoadLibrary("qtdll.dll");
  17. if (hDll != NULL)
  18. {
  19. //定义回调函数指针
  20. typedef void (CALLBACK *PFunCallBack)(FunCallBack);
  21. //实例化回调函数指针
  22. PFunCallBack SetFunCallBack=(PFunCallBack)GetProcAddress(hDll,"SetFunCallBack");
  23. //执行回调函数
  24. if (SetFunCallBack){
  25. SetFunCallBack(HandleEvent);
  26. }
  27. //实例化函数指针
  28. Add add = (Add)GetProcAddress(hDll,"add");
  29. //调用DLL中的方法
  30. printf("%d\n", add(1));
  31. //调用DLL中的方法并触发事件
  32. printf("%d\n", add(110));
  33. FreeLibrary(hDll);
  34. }
  35. return 0;
  36. }




运行结果如图:


当传入的参数为110时,会触发事件,返回值120

第五步:编写C#程序测试
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Runtime.InteropServices;
  5. namespace qtdllTestNET
  6. {
  7.     class Program
  8.     {
  9.         //声明DLL中的函数
  10.         [DllImport("qtdll.dll")]
  11.         public static extern int add(int i);
  12.         //声明DLL中的回调函数,即事件
  13.         [DllImport("qtdll.dll")]
  14.         public static extern void SetFunCallBack([MarshalAs(UnmanagedType.FunctionPtr)] CallbackFun pCallbackFun);        
  15.         [UnmanagedFunctionPointer(CallingConvention.Cdecl)]
  16.         public delegate void CallbackFun(int i);
  17.         public static void HandleEvent(int i)
  18.         {
  19.             Console.WriteLine(i);
  20.         }
  21.         static void Main(string[] args)
  22.         {
  23.             CallbackFun HandleEventX = HandleEvent;
  24.             SetFunCallBack(HandleEventX);            
  25.             Console.WriteLine(add(1));
  26.             Console.WriteLine(add(110));
  27.             Console.ReadLine();
  28.         }
  29.     }
  30. }



运行结果:


CSDN下载地址:http://download.csdn.net/detail/feiyangqingyun/7327095
欢迎关注微信公众号:Qt实战 (各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发)QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线liudianwu

只看该作者 1楼 发表于: 2014-05-10
本站下载地址: QTDLL.zip (5341 K) 下载次数:652
欢迎关注微信公众号:Qt实战 (各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发)QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线wjian10

只看该作者 2楼 发表于: 2014-05-10
很好,正急需呢,感谢。
离线realfan

只看该作者 3楼 发表于: 2014-05-10
这个要赞一个
离线jiriken

只看该作者 4楼 发表于: 2014-05-10
    
离线xianqingzh

只看该作者 5楼 发表于: 2014-05-10
这东西不是到处都有一堆吗?
离线自强不吸

只看该作者 6楼 发表于: 2014-05-11
前面铺垫有点长
自强不吸!
离线maple918

只看该作者 7楼 发表于: 2014-05-12
感谢楼主的分享,小弟正需要呢!
离线pxiao_xiao

只看该作者 8楼 发表于: 2014-05-12
回 xianqingzh 的帖子
xianqingzh:这东西不是到处都有一堆吗? (2014-05-10 23:11) 

人不也哪都是吗?  什么心态
离线jimmysmax

只看该作者 9楼 发表于: 2014-05-15
mark
离线angelus

只看该作者 10楼 发表于: 2014-08-03
赞一个!
离线hhjj3388

只看该作者 11楼 发表于: 2014-08-12
支持你啊·!!!!
离线junbujianwpl

只看该作者 12楼 发表于: 2014-08-26
顶一个 谢谢 分享
离线donglaile

只看该作者 13楼 发表于: 2015-01-08
感谢分享
离线枫界易城

只看该作者 14楼 发表于: 2015-01-13
离线sam_0522

只看该作者 15楼 发表于: 2015-01-13
感谢分享
离线tjchke

只看该作者 16楼 发表于: 2015-08-09
谢谢分享
离线lyjbbq

只看该作者 17楼 发表于: 2015-08-12
qt-solutions里面就有,大家也可以参考下
离线hunterzf

只看该作者 18楼 发表于: 2015-11-18
很好,急需呢
在线clickto

只看该作者 19楼 发表于: 2016-09-05
马克一下!
离线renwinping

只看该作者 20楼 发表于: 2016-12-16
我也写了一个DLL让VC(非QT工程纯C++)的程序,实现在DLL里把画在Qpixmap上的图像保存为png,  但是有个问题,调用按照Qt的设计约定,所有Qt API都必须在Qt环境里运行,也就是要有 QCoreApplication的实例。所以在DLL的接口中,判断如果有Qapp实例就使用,若无则临时构造一个来绕开这个限制, 实在是别扭有高手知道如何处理么?
离线return

只看该作者 21楼 发表于: 2016-12-16
      
离线xiongyingeng

只看该作者 22楼 发表于: 2016-12-24
离线t1029901995

只看该作者 23楼 发表于: 2017-01-08
赶紧收藏了给
离线stlcours

只看该作者 24楼 发表于: 2017-01-16
回 renwinping 的帖子
renwinping:我也写了一个DLL让VC(非QT工程纯C++)的程序,实现在DLL里把画在Qpixmap上的图像保存为png,  但是有个问题,调用按照Qt的设计约定,所有Qt API都必须在Qt环境里运行,也就是要有 QCoreApplication的实例。所以在DLL的接口中,判断如果有Qapp实例就使用,若无则临时构造 .. (2016-12-16 13:26) 

这不是别扭,这是必须的。Delphi里也有这个问题。
两个著名框架都是这样处理的,所以应该就是只能如此。
离线luanl

只看该作者 25楼 发表于: 2019-05-28
感谢楼主分享,大赞一个。
离线baikaishui

只看该作者 26楼 发表于: 2020-03-28
感谢大佬,每天向刘大佬学习
离线mountainhigh

只看该作者 27楼 发表于: 2021-02-09
MFC+QT,很好,正想这么试试,感谢。
离线ifun888

只看该作者 28楼 发表于: 2021-04-12
想问个问题,我看了下那个XXX__global.h文件,里面好像只是一个宏定义,是不是有了这个宏定义就无需在每个要供外部调用的函数前加上declspec这些字段呢?
快速回复
限100 字节
 
上一个 下一个