C++调用C#之C++DLL调用C# COM控件
1. 新建项目
这里我们使用ATL,来接受C# COM控件向外发送的事件。
2. 初始化ATL
#include "stdafx.h" CComModule _module; BOOL APIENTRY DllMain( HMODULE hModule,
DWORD ul_reason_for_call,
LPVOID lpReserved
)
{
switch (ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
{
_module.Init(NULL, hModule);//初始化
}
break;
case DLL_THREAD_ATTACH:
break;
case DLL_THREAD_DETACH:
break;
case DLL_PROCESS_DETACH:
{
_module.Term();//释放
}
break;
}
return TRUE;
}
3. 定义一个显示UI的接口类
IShowUI.h
#ifdef DLL_SHOW_UI
#define DLL_CLASS _declspec(dllimport)
#else
#define DLL_CLASS _declspec(dllexport)
#endif #ifndef _ISHOW_UI_H_
#define _ISHOW_UI_H_ #include <functional>
#include <atlconv.h> using std::function; class DLL_CLASS IShowUI
{
protected:
IShowUI(); virtual ~IShowUI(); public:
static IShowUI* CreateInstance(); static void DestoryInstance(IShowUI* pIShowUI); public:
virtual void SetCallback(function<void(BSTR)>* pCallback) = 0; virtual void ShowDialog(BSTR bstrDialogType, BSTR bstrParam) = 0;
}; #endif
IShowUI.cpp
#include "IShowUI.h"
#include "ShowUIImpl.h" IShowUI::IShowUI()
{ } IShowUI::~IShowUI()
{ } IShowUI* IShowUI::CreateInstance()
{
IShowUI* pIShowUI = new CShowUIImpl;
return pIShowUI;
} void IShowUI::DestoryInstance(IShowUI* pIShowUI)
{
if (pIShowUI)
{
delete pIShowUI;
}
}
4. 定义一个显示UI的实现类
ShowUIImpl.h
#ifndef _SHOW_UI_IMPL_H_
#define _SHOW_UI_IMPL_H_ #include "IShowUI.h"
#ifdef _DEBUG
#import "../NetActiveX/bin/Debug/NetActiveX.tlb"
#else
#import "../NetActiveX/bin/Release/NetActiveX.tlb"
#endif class CHandleEvent; class CShowUIImpl : public IShowUI
{
public:
CShowUIImpl(); virtual ~CShowUIImpl(); public:
virtual void SetCallback(function<void(BSTR)>* pCallback); virtual void ShowDialog(BSTR bstrDialogType, BSTR bstrParam); private:
function<void(BSTR)>* m_pCallback;
CHandleEvent* m_pHandleEvent;
NetActiveX::IShowDialogPtr m_showDialogPtr;
}; #endif
ShowUIImpl.cpp
#include "stdafx.h"
#include "IShowUI.h"
#include "ShowUIImpl.h"
#include "SystemString.h"
#include <atlcom.h> //处理C# COM控件发送的事件
class CHandleEvent : public IDispEventImpl<0,
CHandleEvent,
&(__uuidof(NetActiveX::IEvent)),
&(__uuidof(NetActiveX::__NetActiveX)),
1,
0>
{
public:
CHandleEvent()
{
m_pCallback = NULL;
} void SetEventCallback(std::function<void(BSTR)>* pCallback)
{
m_pCallback = pCallback;
} STDMETHOD(NotifyEvent)(BSTR bstr)
{
if (m_pCallback)
{
(*m_pCallback)(bstr);
}
return S_OK;
} BEGIN_SINK_MAP(CHandleEvent)
SINK_ENTRY_EX(0, (__uuidof(NetActiveX::IEvent)), 20, NotifyEvent)//事件处理函数,此处的事件ID=20必须和C# NotifyEvent定义的完全一样
END_SINK_MAP() private:
function<void(BSTR)>* m_pCallback;//事件处理回调函数
}; CShowUIImpl::CShowUIImpl()
{
CoInitialize(NULL);
m_pCallback = NULL;
m_pHandleEvent = new CHandleEvent;
m_showDialogPtr = NetActiveX::IShowDialogPtr(__uuidof(NetActiveX::ShowDialogImpl));
} CShowUIImpl::~CShowUIImpl()
{
delete m_pHandleEvent;
m_pHandleEvent = NULL;
CoUninitialize();
} void CShowUIImpl::SetCallback(function<void(BSTR)>* pCallback)
{
m_pCallback = pCallback;
} void CShowUIImpl::ShowDialog(BSTR bstrDialogType, BSTR bstrParam)
{
if (m_showDialogPtr.GetInterfacePtr())
{
CSystemString strDialogType(bstrDialogType);
CSystemString strParam(bstrParam); m_pHandleEvent->SetEventCallback(m_pCallback);//设置事件处理回调函数
m_pHandleEvent->DispEventAdvise(m_showDialogPtr);//注册事件
m_showDialogPtr->ShowDialog(strDialogType.GetBSTR(), strParam.GetBSTR());
m_pHandleEvent->DispEventUnadvise(m_showDialogPtr);//取消事件
}
}
5. 其他辅助类
SystemString.h
#ifndef _SYSTEM_STRING_H_
#define _SYSTEM_STRING_H_ class CSystemString
{
public:
explicit CSystemString(BSTR bstr)
{
if (bstr)
{
m_bstr = SysAllocString(bstr);
}
else
{
m_bstr = NULL;
}
} ~CSystemString()
{
Clear();
} void Clear()
{
if (m_bstr)
{
SysFreeString(m_bstr);
m_bstr = NULL;
}
} BSTR GetBSTR() {return m_bstr;} private:
CSystemString(CSystemString&);
CSystemString& operator =(CSystemString&); private:
BSTR m_bstr;
}; #endif
6. 新建一个C++ win32控制台应用程序,调用DLL显示C#对话框
#include "stdafx.h"
#include <string>
#include <functional>
#define DLL_SHOW_UI
#include "IShowUI.h" using std::string;
using std::wstring; #pragma comment(lib, "DllInterface.lib") class EmployeeData
{
public:
void ParseString(wstring s)
{
int beg = -1;
int end = -1; beg = 0;
end = s.find(L",", beg);
if (end != -1)
{
m_name = s.substr(beg, end - beg);
beg = end + 1;
}
end = s.find(L",", beg);
if (end != -1)
{
m_sex = s.substr(beg, end - beg);
beg = end + 1;
}
end = s.find(L",", beg);
if (end != -1)
{
m_age = _wtoi(s.substr(beg, end - beg).c_str());
beg = end + 1;
}
end = s.find(L",", beg);
if (end != -1)
{
m_phone = s.substr(beg, end - beg);
beg = end + 1;
}
end = s.length();
if (beg < end)
{
m_mobile = s.substr(beg, end - beg);
}
} wstring ToString()
{
wstring s = L"";
wchar_t psz[32] = {0}; s += m_name;
s = s + L"," + m_sex;
s = s + L"," + _itow(m_age, psz, 10);
s = s + L"," + m_phone;
s = s + L"," + m_mobile; return s;
} wstring m_name;
wstring m_sex;
int m_age;
wstring m_phone;
wstring m_mobile;
}; void Callback(BSTR bstr)
{
EmployeeData employee; employee.ParseString(bstr);
} int _tmain(int argc, _TCHAR* argv[])
{
IShowUI* pShowUI = IShowUI::CreateInstance();
EmployeeData employee;
function<void(BSTR)> call = std::bind(&Callback, std::tr1::placeholders::_1); employee.m_name = L"123";
employee.m_sex = L"男";
employee.m_age = 10;
employee.m_phone = L"123456";
employee.m_mobile = L"13245678952"; BSTR bstr = L"EmployeeDialog";
wstring s = employee.ToString();
BSTR bstrParam = (BSTR)s.c_str(); pShowUI->SetCallback(&call);//设置事件响应回调
pShowUI->ShowDialog(bstr, bstrParam);//显示员工属性对话框 IShowUI::DestoryInstance(pShowUI);
pShowUI = NULL; return 0;
}
C++调用C#之C++DLL调用C# COM控件的更多相关文章
- c#调用c++制作的基于mfc的ocx控件
原文:http://blog.csdn.net/yhhyhhyhhyhh/article/details/51286926 原文中有问题部分已修改. c#调用c++制作的基于mfc的ocx控件 ...
- Ruby操作VBA的注意事项和技巧(2):宏里调用和控制窗体以及窗体上的控件、不同workbook之间的宏互相调用
4.宏里调用并控制窗体以及窗体上的各种控件 1 Sub Criterion_Check() 2 If Workbooks.count = 0 Then '如果当前没有打开的工作薄的话需要发出警告 3 ...
- Delphi 在DLL中使用DevExpress控件时出错解决办法
测试环境 DevExpress VCL 14.1.3 和XE7 问题:在dll使用cxGrid控件时 如果不添加列标题 则不报错 查询无数据集显示,如果加上标题 就报错了 这段为报错部分 fun ...
- Android调用相机拍摄照片并显示到 ImageView控件中
在前面的一篇文章中曾介绍过简单的开启相机照相功能,详见 Android简单调用相机Camera功能,实现打开照相功能 ,这一次就会将前面拍摄的照片显示到ImageView中,形成一个完整的效果 看实例 ...
- C#调用ActiveX控件
背景:最近项目中需要用到ActiveX控件,项目是在.Net平台下开发的.因此就直接在项目中添加了对ActiveX控件的引用,添加引用成功.在代码中实例化类的实例也没有问题,但在调用其方法或属性时总是 ...
- Java 通过 jacob调用OCX控件
安装好要调用的ocx驱动,并通过注册表查询其对应的clsid. 下载jacob-1.18解压,将jacob-1.18-x86.dll文件复制到D:\Program Files (x86)\Java\j ...
- C#调用第三方ocx控件 (winform /aspx)
C#调用第三方ocx控件 1..net环境在工具箱上点右键,选择自定义工具箱,然后选择你需要的COM或者OCX控件就可以了. 2.在自定义工具箱中加入相应的控件,设置id,在客户端脚本中直接引用它 ...
- Qt开发Activex笔记(二):Qt调用Qt开发的Activex控件
若该文为原创文章,转载请注明原文出处本文章博客地址:https://blog.csdn.net/qq21497936/article/details/113789693 长期持续带来更多项目与技术分享 ...
- VS2010 开发 VB6.0 activeX控件 dll
项目源码 https://download.csdn.net/download/csdn_z_s/10427764 开发环境 操作系统: win7 64位 旗舰版 Java语言开发环境: Eclip ...
随机推荐
- Linux中kettle自动化部署脚本
自己写的一个自动化在Linux中部署kettle的脚本,包括一些遇到的问题在脚本中都有涉及. kettle是官网最新版本pdi-ce-6.1.0.1-196.zip 目前最新版本下载地址:https: ...
- c++的函数模板和类模板
函数模板和普通函数区别结论: 函数模板不允许自动类型转化 普通函数能够进行自动类型转换 函数模板和普通函数在一起,调用规则: 1 函数模板可以像普通函数一样被重载 2 C++编译器优先考虑普通函数 3 ...
- 深度探索C++对象模型之C++对象模型笔记
0.菜鸟觉得,在看这本书的时候最好切换角色,把自己的思维转换成编译器开发者,去考虑问题,这样会容易理解些.(当然这样很难,就想着自己要解决什么样的问题好了) 1.在C++中,类的数据成员有两种:静态和 ...
- java读写串口
http://blog.csdn.net/xxyy888/article/details/8946046
- java反射机制(2)
首先,我们在开始前提出一个问题: 1.在运行时,对于一个java类,能否知道属性和方法:能否去调用它的任意方法? 答案是肯定的. 本节所有目录如下: 什么是JAVA的反射机制 JDK中提供的Refle ...
- 菲菲更名宝贝 得意非凡版 v1.9 免费绿色版
软件名称: 菲菲更名宝贝 得意非凡版软件语言: 简体中文授权方式: 免费软件运行环境: Win8 / Win7 / Vista / WinXP软件大小: 12.5MB图片预览: 软件简介:菲菲更名宝贝 ...
- 如何成为出色的IT项目经理:成功的五个关键因素
“出色”的IT 项目经理的定义不是一成不变的.随着经济和商业因素的改变,项目经理的角色进行调整以适应新的需求,迎接新的挑战. 除了一般的困惑之外,还有一种看法就是,在组织中,不同的人对于项目经理的看法 ...
- Understanding Neural Networks Through Deep Visualization
当数据一层一层通过更多的卷积层时,你可以得到的特征图像代表的特征就会更加的复杂. 在网络的最后,你也许可以得到一个抽象的物体.如果你想通过可视化方法在卷积神经网络中看到更多的信息.这里有一个工具方便你 ...
- 不合法语句 self.contentView.frame.origin.x = x;
下面的写法是错误的: CGFloat x = self.contentView.frame.origin.x; x = _lastDownX + translation.x; self.content ...
- arguments对象,caller 和 callee
arguments对象是比较特别的一个对象,arguments非常类似Array,但实际上又不是一个Array实例. 它指的是函数对象里的参数,且只能在函数内部使用. 使用 检测函数的参数个数,引用属 ...