VC 怎样调用COM控件的接口函数

来源:互联网 发布:淘宝店运营模式 编辑:程序博客网 时间:2024/05/20 23:35

--------------------------------
COM库函数
--------------------------------
利用COM库函数使用代码组件的方法是本文介绍的三种方法中实现起来最麻烦和困难的方法。它要求开发人员必须具有对COM原理的深入理解。该方法实现步骤如下:
1. 首先添加COM初始和终止代码。在应用程序类的初始化实例函数InitInstance()中添加如下代码:
CoInitialize(NULL);
……
CoUnInitialize();
上述语句运行在MFC框架/非MFC框架中,但由于本文程序使用MFC框架,所以也可以利用AfxOleInit()函数对其进行初始化。
2. 然后用#include 语句包含对组件头文件的引用并创建组件对象。在头文件中包含了接口的C++定义以及说明接口ID IID和类ID CLSID的符号化常量。创建工作在初始化对话框函数中进行:
IAccount pAccount=NULL;
……
CoCreateInstance(CLSID_Account,
NULL,
CLSCTX_INPROC_SERVER,
IID_Iaccount,
reinterpret_cast (&pAccount));
3. 最后释放组件对象。该工作应在程序退出之前完成,比如在消息WM_CLOSE的响应函数中进行:
if(pAccount!=NULL)
pAccount->Realease();
对该代码组件中的其他功能函数的调用,可以通过组件对象的接口指针pAccount来进行:
……
BSTR bstrResult;
PAccount-> Post(100,bstrResult);
SysFreeString(bstrResult);
……
由于COM支持类在comdef. h中定义,所以还要包含对该头文件的引用,才可以使程序正常运行。
-------------------------------- 类向导
--------------------------------
通过类向导可以直接阅读组件的类型库,并产生包装类型库中每个接口的类,通过这些类的成员函数可以访问组件接口的方法和属性,与使用ActiveX控件的方法有些类似。
首先添加对COM组件进行初始化的代码。我们可以通过类向导的From a Type Library加入组件的.tlb类型库文件,并从中引入其接口类。在本例中引入的类型库文件中只包含一个从ColeDispatchDriver派生的组件包装类IAccount。通过包装类的成员,可以了解到组件接口能提供哪些服务,而且可以通过它们访问组件接口的方法和属性。
在初始化对话框函数里用COleDispatchDriver类的CreateDispatch()成员函数创建Account组件对象:
IAccount m_account;
……
m_account.CreateDispatch(“ATLSample.Account.1”));
其中ProgID值“ATLSample. Account. 1”可以通过Microsoft Visual Studio Tools 6.0里的OLE View工具查找到,其前提是该组件已被成功注册过。
释放Account组件对象也可以用COleDispatch-Driver类的ReleaseDispatch()函数来完成。
对于在COM库函数方法中用过的Post方法可用下述方法调用:
CString str=m_account. Post(100);
可以看出此种方法实现了同样的功能但实现起来要比上一种方法简单些,而且对理解COM的要求也不高。
--------------------------------
#import 指令
--------------------------------
#import 指令方法非常简便。对于类型库文件采用该指令是非常合适的,因为不管是调试版本还是发行版本,对于类型库文件而言,其路径是固定的。#import指令在执行时将会从待引入的类型库中提取出两个文件:一个.tlh文件和一个.tli文件,后者仅仅是包装类的函数实现,而前者则包含了许多有关的重要信息。智能接口指针也在其中定义:
_COM_SMARTPTR_TYPEDEF(IAccount,__uuidof(IAccount));
在实际编译时,编译器会将其展开成下述代码,并通过_com_ptr_t模板类为接口IAccount定义一个智能指针IAccountPtr。之所以说其是智能指针,是由于它替代IAccount时,会自动处理CoCreate-Instance和所有的IUnknow方法,使用起来非常方便:
typedef _com_ptr_t<_com_IIID< Iaccount,__uuidof(IAccount)>> IAccountPtr;
由于有了智能指针,我们就可以调用_com_ptr_t模板类的CreateInstance()函数来完成对接口指针的创建工作:
IAccountPtr m_account=NULL;
m_account.CreateInstance(__uuidof(Account));
由于在生成的.tlh文件中包含结构声明和declspec(uuid(“”))声明,所以在这里可以很方便地用__uuidof(Account)获取接口的GUID。declspec(uuid(“”))声明将GUID和类及每个接口联系起来,允许开发人员以uuidof操作符来获取类和接口的GUID。
需要特别指出的是: 为防止原有代码和新引入的代码之间发生名字冲突,编译器会定义一个由类型库名称标识的命名空间,并在其中声明的任何名称内附加一个标识符。而为了避免指定命名空间标识,可以在#import 语句后加上using namespace,而且还可以用rename_namespace来改变命名空间。比如在本例中可以进行如下处理:
#import “Account.tlb” rename_namespace(“AccountDriver”)
using namespace AccountDriver;
这样,在使用智能接口指针IAccountPtr时只需定义即可:
IAccountPtr m_account;
至于对代码组件中的函数和属性的调用则同前两种方法一样,也是通过m_account来完成访问的。由于_com_ptr_t模板类和智能指针的引入,#import 指令方法是这三种方法中使用COM组件最简单的一种。

----------------------------------------------------------------------------

----------------------------------------------------------------------------

准备及条件:
COM服务器为进程内服务器,DLL名为simpCOM.dll,该组件只有一个接口IFoo,该接口只有一个方法HRESULT SayHello(void)
在SDK中调用
=====================================
一、最简单最常用的一种,用#import导入类型库,利用VC提供的智能指针包装类
演示代码:     
#import "D:/Temp/vc/simpCOM/Debug/simpCOM.dll" no_namespace
CoInitialize(NULL);
{
IFooPtr spFoo = NULL;
spFoo.CreateInstance(__uuidof(Foo));
spFoo->SayHello();
}
CoUninitialize();     
二、引入midl.exe产生的*.h,*_i.c文件,利用CoCreateInstance函数来调用
演示代码:
/*在工程中加入*_i.c文件,例如本例的simpCOM_i.c,该文件定义了类和接口的guid值,如果不引入的话,会发生连接错误。*/
#include "D:/Temp/vc/simpCOM/simpCOM.h"
CoInitialize(NULL);
IFoo* pFoo = NULL;
HRESULT hr = CoCreateInstance(CLSID_Foo, NULL, CLSCTX_ALL, IID_IFoo, (void**)&pFoo);
if (SUCCEEDED(hr) && (pFoo != NULL))
{
pFoo->SayHello();
pFoo->Release();
}
CoUninitialize();
三、不用CoCreateInstance,直接用CoGetClassObejct得到类厂对象接口,然后用该接口的方法CreateInstance来生成实例。
演示代码:
/*前期准备如二方法所述*/
IClassFactory* pcf = NULL;
HRESULT hr = CoGetClassObject(CLSID_Foo, CLSCTX_ALL, NULL, IID_IClassFactory, (void**)&pcf);
if (SUCCEEDED(hr) && (pcf != NULL))
{
IFoo* pFoo = NULL;
hr = pcf->CreateInstance(NULL, IID_IFoo, (void**)&pFoo);
if (SUCCEEDED(hr) && (pFoo != NULL))
{
pFoo->SayHello();
pFoo->Release();
}
pcf->Release();
}
四、不用CoCreateInstance or CoGetClassObject,直接从dll中得到DllGetClassObject,接着生成类对象及类实例(本方法适合于你想用某个组件,却不想在注册表中注册该组件)
演示代码:
/*前期准备工作如二方法所述,事实上只要得到CLSID和IID的定义及接口的定义就行*/
typedef HRESULT (__stdcall * pfnGCO) (REFCLSID, REFIID, void**);
pfnGCO fnGCO = NULL;
HINSTANCE hdllInst = LoadLibrary("D://Temp//vc//simpCOM//Debug//simpCOM.dll");
fnGCO = (pfnGCO)GetProcAddress(hdllInst, "DllGetClassObject");
if (fnGCO != 0)
{
IClassFactory* pcf = NULL;
HRESULT hr=(fnGCO)(CLSID_Foo, IID_IClassFactory, (void**)&pcf);
if (SUCCEEDED(hr) && (pcf != NULL))
{
IFoo* pFoo = NULL;
hr = pcf->CreateInstance(NULL, IID_IFoo, (void**)&pFoo);
if (SUCCEEDED(hr) && (pFoo != NULL))
{
pFoo->SayHello();
pFoo->Release();
}
pcf->Release();
}
}
FreeLibrary(hdllInst);
在MFC中调用
=====================================
在MFC中除了上面的几种方法外,还有一种更方便的方法,就是通过ClassWizard利用类型库生成包装类,不过有个前提就是com组件的接口必须是派生自IDispatch
具体方法:
1、按Ctrl+W调出类向导,按Add Class按钮弹出新菜单,选From a type libarary,然后定位到simpCOM.dll,接下来会出来该simpCOM中的所有接口,选择你想生成的接口包装类后,向导会自动生成相应的.cpp和.h文件.
这样你就可以在你的MFC工程中像使用普通类那样使用COM组件了.
演示代码:
CoInitialize(NULL);
IFoo foo;
if (foo.CreateDispatch("simpCOM.Foo") != 0)
{
foo.SayHello();
foo.ReleaseDispatch();
}
CoUninitialize();

----------------------------------------------------------------------

在VC程序中调用服务组件
1、运行VC 6.0,使用MFC程序向导创建一个新的基于对话框的MFC应用程序。
2、使用#import导入服务组件,拷贝PRJADO.DLL到VC工程目录,打开STDAFX.H文件增加如下代码:
# import "prjado.dll"
using namespace prjado;
这段代码应该放置在:
//{{AFX_INSERT_LOCATION}}
// Microsoft Visual C++ will insert additional declarations immediately 之后和#endif 之前。
在stdafx.h文件导入dll能够让编译器在运行时连接dll的类型库,#import它能够自动产生一个对GUIDs的定义,同时自动生成对clsado对象的封装。同时能够列举它在类中所能找到的类型, VC++会在编译的时候自动生成两个文件:
一个头文件(.tlh),它包含了列举的类型和对类型库中对象的定义;
一个实现文件(.tli)对类型库对象模型中的方法产生封装。
Namespace(名字空间)用来定义一个名字空间,使用unsing就可以将当前的类型上下文转换名字空间所定地,让我们可以访问服务组件的方法。
如果我们修改了服务组件程序,建议删除这两个文件后重新完整编译工程,以便让编译器重新列举类的属性以及函数。
3、在对话框中新添加一按钮,标题设为“测试”。双击“测试”按钮,系统提示新建一函数OnButton1对应于该按钮的CLICK事件。OnButton1函数的具体代码如下
HRESULT hresult;
CLSID clsid;
CoInitialize(NULL);
//初始化COM接口
hresult=CLSIDFromProgID(OLESTR("prjado.clsado"), &clsid);
//得到组件的CLSID
_clsado *t;
//这就是我们在PRJADO.DLL中定义的类
hresult=CoCreateInstance(clsid,NULL,CLSCTX_INPROC_SERVER,__uuidof(_clsado),(LPVOID *) &t);
初始化CLSADO,CoCreateInstance()的第一个参数是对象的CLSID(类代码),这个128位的编码唯一地标识了组件服务,它的路径和运行参数等信息都放在系统注册表中;第二个参数一般情况下设为NULL;第三个参数用来指明对象的运行环境,如远程或本地,此处设为CLSCTX_INPROC_SERVER,标明是服务组件是一个DLL;第四个参数指明用来与对象通信接口的ID,这也是一个128位的编码;第五个参数是用来接收对象的接口指针。

if(FAILED(hresult))
{
AfxMessageBox("初始化服务组件失败");
return;
}
//如果初始化失败,弹出提示对话框后退出程序
t->ado_Init() ;
//调用初始化函数
t->ado_add("从VC调用组件") ;
//调用增加记录函数
t->ado_UnInit ();
//调用关闭连接函数
t->Release();
//释放接口指针
CoUninitialize();
// 关闭COM库的联接

AfxMessageBox("调用服务组件完成");
保存工程后编译运行VC 程序,点击“测试”按钮,等待完成提示后,我们打开数据库就可以看到刚才添加的记录了。  

原创粉丝点击