zoukankan      html  css  js  c++  java
  • 在VC中调用COM组件的方法

     

    整理自网络,仅供参考: http://bbs.csdn.net/topics/50319093

     

    调用环境:
    COM
    服务器为进程内服务器,DLL名为simpCOM.dll,该组件只有一个接口IFoo,该接口只有一个方法HRESULT SayHello(void)


    SDK中调用
    =====================================
    方法一:最简单最常用的一种,用#import导入类型库,利用VC提供的智能指针包装类


    演示代码:

    #import "D:TempvcsimpCOMDebugsimpCOM.dll" no_namespace


    CoInitialize(NULL);
    //
    IFooPtr spFoo = NULL;
    spFoo.CreateInstance(__uuidof(Foo));
    spFoo->SayHello();
    spFoo.Release();/*
    晕死了,本来智能指针就是为了让用户不用关心这个的,可是我发现如果不手工调用一下的话,程序退出后会发生内存访问错误,我是在console中做试验的,哪位大侠知道怎么回事请一定指教?答:spFoo智能指针对象在CoUninitialize()函数之后析构引起*/

    //
    CoUninitialize();

        也可以: 

        ::CoInitialize( NULL ); // 在这里进行 COM 初始化,要注意智能指针的释放

         //
        CComQIPtr <IFoo> spFoo;
        HRESULT hr = spFoo.CoCreateInstance( __uuidof(Foo));
        spFun->Foo();
        spFun.Release();  // 错误:spFun->Release();
        //
        ::CoUninitialize();

     

    也可以这么创建接口:

    hr=CoCreateInstance(__uuidof(atlTE3Lib::NewObject),NULL,CLSCTX_INPROC_SERVER, __uuidof(atlTE3Lib::INewObject), (void **)&m_pNewObject);

    // 如果没有指明"no_namespace",得指明命名空间" atlTE3Lib"。注意替换类名及接口名称。

    方法二:引入midl.exe产生的*_i.h,*_i.c文件,利用CoCreateInstance函数来调用

    演示代码:


    /*
    在工程中加入*_i.c文件,例如本例的simpCOM_i.c,该文件定义了类和接口的guid值,如果不引入的话,会发生连接错误。*/

    #include "D:TempvcsimpCOMsimpCOM_i.h"
    #include "D:TempvcsimpCOMsimpCOM_i.c"


    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,接着生成类对象及类实例(本方法适合于你想用某个组件,却不想在注册表中注册该组件)

    演示代码:


    /*
    前期准备工作如二方法所述,事实上只要得到CLSIDIID的定义及接口的定义就行*/
    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。(为什么必须派生自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();

     

  • 相关阅读:
    数据结构——数据结构的起源和研究内容
    数据结构——学习数据结构的意义
    C++中动态内存申请的结果
    C++中函数异常规格的说明
    C++异常处理的深入理解
    NOIP 2012 Day2
    NOIP 2012 Day1
    NOIP 2011 Day2
    NOIP 2011 Day 1
    NOIP 2010
  • 原文地址:https://www.cnblogs.com/ant-wjf/p/3201650.html
Copyright © 2011-2022 走看看