C++调用Com
需求:
1.创建myCom.dll,该COM只有一个组件,两个接口:
IGetRes--方法Hello(),
IGetResEx--方法HelloEx()
2.在工程中导入组件或类型库
#import "组件所在目录myCom.dll" no_namespace
或
#import "类型库所在目录myCom.tlb"
using namespace MYCOM;
方法一:
- CoInitialize(NULL);
- CLSID clsid;
- CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- CComPtr<IGetRes> pGetRes;//智能指针
- pGetRes.CoCreateInstance(clsid);
- pGetRes->Hello();
- pGetRes.Release();//小心哦!!请看最后的“注意”
- CoUninitialize();
- CoInitialize(NULL);
- CLSID clsid;
- CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- CComPtr<IGetRes> pGetRes;//智能指针
- pGetRes.CoCreateInstance(clsid);
- pGetRes->Hello();
- pGetRes.Release();//小心哦!!请看最后的“注意”
- CoUninitialize();
方法二:
- CoInitialize(NULL);
- CLSID clsid;
- HRESULT hr=CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- IGetRes *ptr;
- hr=CoCreateInstance(clsid,NULL,CLSCTX_INPROC_SERVER,
- __uuidof(IGetRes),(LPVOID*)&ptr);
- ptr->Hello();
- p_image001 CoUninitialize();
- CoInitialize(NULL);
- CLSID clsid;
- HRESULT hr=CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- IGetRes *ptr;
- hr=CoCreateInstance(clsid,NULL,CLSCTX_INPROC_SERVER,
- __uuidof(IGetRes),(LPVOID*)&ptr);
- ptr->Hello();
- p_image001 CoUninitialize();
方法三:
- CoInitialize(NULL);
- HRESULT hr;
- CLSID clsid;
- hr=CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- IGetRes* ptr;
- IGetResEx* ptrEx;
- //使用CoCreateClassObject创建一个组件(特别是mutilThreads)的多个对象的
- 时候,效率更高.
- IClassFactory* p_classfactory;
- hr=CoGetClassObject(clsid,CLSCTX_INPROC_SERVER,
- NULL,IID_IClassFactory,
- (LPVOID*)&p_classfactory);
- p_classfactory->CreateInstance(NULL,__uuidof(IGetRes),
- (LPVOID*)&ptr);
- p_classfactory->CreateInstance(NULL,__uuidof(IGetResEx),
- (LPVOID*)&ptrEx);
- ptr->Hello();
- ptrEx->HelloEx();
- CoInitialize(NULL);
- HRESULT hr;
- CLSID clsid;
- hr=CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- IGetRes* ptr;
- IGetResEx* ptrEx;
- //使用CoCreateClassObject创建一个组件(特别是mutilThreads)的多个对象的
- 时候,效率更高.
- IClassFactory* p_classfactory;
- hr=CoGetClassObject(clsid,CLSCTX_INPROC_SERVER,
- NULL,IID_IClassFactory,
- (LPVOID*)&p_classfactory);
- p_classfactory->CreateInstance(NULL,__uuidof(IGetRes),
- (LPVOID*)&ptr);
- p_classfactory->CreateInstance(NULL,__uuidof(IGetResEx),
- (LPVOID*)&ptrEx);
- ptr->Hello();
- ptrEx->HelloEx();
方法四:
直接从dll中得到DllGetClassObject,接着生成类对象及类实例(这方法可以使组件不用在注册表里注册,这是最原始的方法,但这样做没什么意义,至少失去了COM对用户的透明性),不推荐使用.
- typedef HRESULT (__stdcall * pfnHello)(REFCLSID,REFIID,void**);
- pfnHello fnHello= NULL;
- HINSTANCE hdllInst = LoadLibrary("组件所在目录myCom.dll");
- fnHello=(pfnHello)GetProcAddress(hdllInst,"DllGetClassObject");
- if (fnHello != 0)
- {
- IClassFactory* pcf = NULL;
- HRESULT hr=(fnHello)(CLSID_GetRes,IID_IClassFactory,(void**)&pcf);
- if (SUCCEEDED(hr) && (pcf != NULL))
- {
- IGetRes* pGetRes = NULL;
- hr = pcf->CreateInstance(NULL, IID_IFoo, (void**)&pGetRes);
- if (SUCCEEDED(hr) && (pFoo != NULL))
- {
- pGetRes->Hello();
- pGetRes->Release();
- }
- pcf->Release();
- }
- }
- FreeLibrary(hdllInst);
- typedef HRESULT (__stdcall * pfnHello)(REFCLSID,REFIID,void**);
- pfnHello fnHello= NULL;
- HINSTANCE hdllInst = LoadLibrary("组件所在目录myCom.dll");
- fnHello=(pfnHello)GetProcAddress(hdllInst,"DllGetClassObject");
- if (fnHello != 0)
- {
- IClassFactory* pcf = NULL;
- HRESULT hr=(fnHello)(CLSID_GetRes,IID_IClassFactory,(void**)&pcf);
- if (SUCCEEDED(hr) && (pcf != NULL))
- {
- IGetRes* pGetRes = NULL;
- hr = pcf->CreateInstance(NULL, IID_IFoo, (void**)&pGetRes);
- if (SUCCEEDED(hr) && (pFoo != NULL))
- {
- pGetRes->Hello();
- pGetRes->Release();
- }
- pcf->Release();
- }
- }
- FreeLibrary(hdllInst);
方法五:
通过ClassWizard利用类型库生成包装类,不过前提是com组件的接口必须是派生自IDispatch,具体方法:
调出添加类向导(.NET中),选择类型库中MFC类,打开,选择"文件",选择"myCom.dll"或"myCom.tlb",接下来会出来该myCom中的所有接口,选择你想生成的接口包装类后,向导会自动生成相应的.h文件.这样你就可以在你的MFC中像使用普通类那样使用组件了.(CreateDispatch("myCom.GetRes") 中的参数就是ProgID通过Clsid在注册表中可以查询的到)
- CoInitialize(NULL);
- CGetRes getRest;
- if (getRest.CreateDispatch("myCom.GetRes") != 0)
- {
- getRest.Hello();
- getRest.ReleaseDispatch();
- }
- CoUninitialize();
- CoInitialize(NULL);
- CGetRes getRest;
- if (getRest.CreateDispatch("myCom.GetRes") != 0)
- {
- getRest.Hello();
- getRest.ReleaseDispatch();
- }
- CoUninitialize();
注意:
COM中的智能指针实际上是重载了->的类,目的是为了简化引用记数,几不需要程序员显示的调用AddRef()和Release(),但是为什么我们在Method 1中pGetRes.Release(),问题在与,我们的智能指针pGetRes生命周期的结束是在CoUninitialize()之后,CoInitialize所开的套间在CoUninitialize()后已经被关闭,而pGetRes此时发生析构,导致了程序的崩溃,解决这个问题的另一个方法是
- CoInitialize(NULL);
- CLSID clsid;
- CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- {
- CComPtr<IGetRes> pGetRes;//智能指针
- pGetRes.CoCreateInstance(clsid);
- pGetRes->Hello();
- }
- CoUninitialize();
- CoInitialize(NULL);
- CLSID clsid;
- CLSIDFromProgID(OLESTR("myCom.GetRes"),&clsid);
- {
- CComPtr<IGetRes> pGetRes;//智能指针
- pGetRes.CoCreateInstance(clsid);
- pGetRes->Hello();
- }
- CoUninitialize();
以上就是COM的5中方法,当然具体怎么使用还是在于程序的环境,加以琢磨....
C++调用Com的更多相关文章
- 黑马毕向东Java基础知识总结
Java基础知识总结(超级经典) 转自:百度文库 黑马毕向东JAVA基础总结笔记 侵删! 写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部 ...
- 《果壳中的C# C# 5.0 权威指南》 - 学习笔记
<果壳中的C# C# 5.0 权威指南> ========== ========== ==========[作者] (美) Joseph Albahari (美) Ben Albahari ...
- JS调用Android、Ios原生控件
在上一篇博客中已经和大家聊了,关于JS与Android.Ios原生控件之间相互通信的详细代码实现,今天我们一起聊一下JS调用Android.Ios通信的相同点和不同点,以便帮助我们在进行混合式开发时, ...
- 【原创分享·支付宝支付】HBuilder打包APP调用支付宝客户端支付
前言 最近有点空余时间,所以,就研究了一下APP支付.前面很早就搞完APP的微信支付了,但是由于时间上和应用上的情况,支付宝一直没空去研究.然后等我空了的时候,发现支付宝居然升级了支付逻辑,虽然目前还 ...
- 操作系统篇-调用门与特权级(CPL、DPL和RPL)
|| 版权声明:本文为博主原创文章,未经博主允许不得转载. 一.前言 在前两篇文章(<操作系统篇-浅谈实模式与保护模式>和<操作系统篇-分段机制与GDT|LDT>)中,我们提到 ...
- 架构设计:远程调用服务架构设计及zookeeper技术详解(下篇)
一.下篇开头的废话 终于开写下篇了,这也是我写远程调用框架的第三篇文章,前两篇都被博客园作为[编辑推荐]的文章,很兴奋哦,嘿嘿~~~~,本人是个很臭美的人,一定得要截图为证: 今天是2014年的第一天 ...
- django server之间通过remote user 相互调用
首先,场景是这样的:存在两个django web应用,并且两个应用存在一定的联系.某些情况下彼此需要获取对方的数据. 但是我们的应用肯经都会有对应的鉴权机制.不会让人家随随便便就访问的对吧.好比上车要 ...
- 调用AJAX做登陆和注册
先建立一个页面来检测一下我们建立的用户名能不能用,看一下有没有已经存在的用户名吗 可以通过ajax提示一下 $("#uid").blur(function(){ //取用户名 va ...
- nodejs进阶(2)—函数模块调用
函数调用 1. 文件内普通函数调用 创建一个js文件命名为2_callFunction.js,其中定义一个函数fun1,向返回对象输出了一段字符串“你好,我是fun1”. //------------ ...
- 【初学python】使用python调用monkey测试
目前公司主要开发安卓平台的APP,平时测试经常需要使用monkey测试,所以尝试了下用python调用monkey,代码如下: import os apk = {'j': 'com.***.test1 ...
随机推荐
- 多个client与一个server端通信的问题
多个client与一个server端通信的问题 上篇博文主要是讲的关于client与server端的通信问题.在上篇博文中当我们仅仅有一个client訪问我们的server时是能够正常执行的,可是当我 ...
- 通过uri呼起本地app
1.在Android本地app清单文件里配置 <activity android:name="com.mdj.ui.WelcomeActivity" android:scre ...
- BZOJ1179 : [Apio2009]Atm 缩点+spfa
1179: [Apio2009]Atm Time Limit: 15 Sec Memory Limit: 162 MBSubmit: 2069 Solved: 826[Submit][Status ...
- Codevs 1744 格子染色==BZOJ 1296 粉刷匠
1744 格子染色 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 题目描述 Description 有 n 条木板需要被粉 ...
- Node.js+Express搭建博客系统基本环境安装
1.下载安装node.js 官网下载地址:https://nodejs.org/en/download/ 2.安装express. 打开node命令行工具,在命令行中输入:npm install -g ...
- 7-80 HTML5新增的JS选择器
7-80 HTML5新增的JS选择器 学习要点 HTML5新增的JS选择器 在传统的 JavaScript 开发中,原生的 JavaScript 所提供的 DOM 选择方法并不多,仅仅局限于通过 ta ...
- 基于Jquery插件Uploadify实现实时显示进度条上传图片
网址:http://www.jb51.net/article/83811.htm 这篇文章主要介绍了基于Jquery插件Uploadify实现实时显示进度条上传图片的相关资料,感兴趣的小伙伴们 ...
- Tensorflow学习笔记——占位符和feed_dict(二)
创建了各种形式的常量和变量后,但TensorFlow 同样还支持占位符.占位符并没有初始值,它只会分配必要的内存.在会话中,占位符可以使用 feed_dict 馈送数据. feed_dict是一个字典 ...
- werkzeug 详解
首先,先向大家介绍一下什么是 werkzeug,Werkzeug是一个WSGI工具包,他可以作为一个Web框架的底层库.这里稍微说一下, werkzeug 不是一个web服务器,也不是一个web框架, ...
- CentOS 6.5克隆后eth1与eth0的问题
CentOS 6.5克隆后eth1与eth0的问题 按照安装文档执行以下步骤时: 从克隆出来的虚拟机网卡都会被命名为eth1,而有些程序或者脚本,涉及到网卡的,默认写的是eth0,这时就存在要 ...