由于原文章沒(méi)有轉(zhuǎn)載鏈接,只好直接粘貼過(guò)來(lái),附上原文地址:http://www.cnblogs.com/feiyangqingyun/p/3720777.html
既然QT也是C++,而且有個(gè)大名鼎鼎的WPS都是QT和MFC混合編程的,所以肯定有方法的,百度谷歌,找到一個(gè)qtwinmigrate的東西,在這里萬(wàn)分感謝qtwinmigrate的作者!
第一步:
打開(kāi)qt?creator,新建C++?庫(kù)項(xiàng)目
依次下一步,記得在選擇需要的模塊的時(shí)候選擇QtGui模塊,如果沒(méi)有選中的話(huà),后面編譯通不過(guò),qmfcapp這個(gè)類(lèi)里面用了QtGui模塊中的方法。
?
第二步:
將多余的文件刪除,_global.h還有其他兩個(gè)文件刪除,因?yàn)檫@是QT新建DLL自帶的,我們要寫(xiě)的是可以供其他語(yǔ)言調(diào)用的DLL,我這里的例子是只用了一個(gè)函數(shù)和事件,沒(méi)有用到界面的東西,所以從qtwinmigrate下面的src文件夾拷貝qmfcapp.h和qmfcapp.cpp這兩個(gè)文件過(guò)來(lái)即可,如下。
?
?
第三步:
更改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函數(shù)入口
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;
}
//定義函數(shù)回調(diào)指針
typedef void (CALLBACK *FunCallBack)(int par);
//定義回調(diào)函數(shù)
FunCallBack OnEvent=NULL;
//定義回調(diào)參數(shù)
int par;
//定義C語(yǔ)言類(lèi)型導(dǎo)出事件函數(shù)
extern "C" __declspec(dllexport) void SetFunCallBack(FunCallBack fun,int p)
{
OnEvent=fun;
par=p;
}
//定義C語(yǔ)言類(lèi)型導(dǎo)出函數(shù)
extern "C" __declspec(dllexport) int add(int i)
{
//下面這段代碼用來(lái)觸發(fā)事件,如果僅僅是使用add函數(shù)的話(huà)可以刪除.
if (i==110){
if (OnEvent){
//當(dāng)傳入?yún)?shù)為110而且已經(jīng)定義過(guò)回調(diào)函數(shù)的話(huà),則觸發(fā)事件.
OnEvent(120);
}
}
return i*2;
}View Code
選擇release編譯,在目錄下生成了一個(gè)DLL文件,這個(gè)文件就可以供VC/C#/VB/JAVA等調(diào)用了。
?
第四步:編寫(xiě)VC和C語(yǔ)言程序測(cè)試
main.c代碼如下:
#include
#include
//定義函數(shù)指針
typedef int (*Add)(int);
//定義回調(diào)函數(shù)指針
typedef void (CALLBACK *FunCallBack)(int);
//定義回調(diào)函數(shù)處理方法
void CALLBACK HandleEvent(int par)
{
printf("%dn", par);
}
//主函數(shù)入口
int main(int argc, char const *argv[])
{
//定義DLL句柄
HINSTANCE hDll = LoadLibrary("qtdll.dll");
if (hDll != NULL)
{
//定義回調(diào)函數(shù)指針
typedef void (CALLBACK *PFunCallBack)(FunCallBack);
//實(shí)例化回調(diào)函數(shù)指針
PFunCallBack SetFunCallBack=(PFunCallBack)GetProcAddress(hDll,"SetFunCallBack");
//執(zhí)行回調(diào)函數(shù)
if (SetFunCallBack){
SetFunCallBack(HandleEvent);
}
//實(shí)例化函數(shù)指針
Add add = (Add)GetProcAddress(hDll,"add");
//調(diào)用DLL中的方法
printf("%dn", add(1));
//調(diào)用DLL中的方法并觸發(fā)事件
printf("%dn", add(110));
FreeLibrary(hDll);
}
return 0;
}View Code
運(yùn)行結(jié)果如圖:
當(dāng)傳入的參數(shù)為110時(shí),會(huì)觸發(fā)事件,返回值120。
?
第五步:編寫(xiě)C#程序測(cè)試
using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
namespace qtdllTestNET
{
class Program
{
//聲明DLL中的函數(shù)
[DllImport("qtdll.dll")]
public static extern int add(int i);
//聲明DLL中的回調(diào)函數(shù),即事件
[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
運(yùn)行結(jié)果: