程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> VC >> vc教程 >> 在VC++6.0中用MFC進行COM編程

在VC++6.0中用MFC進行COM編程

編輯:vc教程
 首先應當明確,MFC中是通過嵌套類而不是多重繼承來實現COM接口的,通過接口映射機制將接口和實現該接口的嵌套類關聯起來;MFC中提供一套簡明的宏來實現嵌套類的定義.其次,MFC通過CCmdTarget類實現了IUnknown接口。

  本文首先描述創建一個COM服務器的步驟和核心代碼.然後說明客戶程序關鍵代碼。

  此COM服務器實現一個TimeLogServer組件,為簡明起見,此組件只有一個接口ITimeLog,通過ITimeLog的方法OutputLog可以將日志文本輸出至日志文件。

  創建一個MFC DLL工程,選擇支持Automation(當然本程序不一定是自動化服務器,在這裡這樣做好處在於自動實現了幾個必要的輸出函數如DllGetClassObject,DllRegisterServer等,否則要自己寫)

  第一節 COM服務器

  一. 聲明組件和接口

  1.寫一個GUIDs.h,在GUIDs.h中聲明組件和接口的GUID

//聲明組件GUID {A433E701-E45E-11d3-97B5-52544CBA7F28}
//DEFINE_GUID(CLSID_TimeLogServer,
//0xa433e701, 0xe45e, 0x11d3, 0x97, 0xb5, 0x52, 0x54, 0x4c, 0xba, 0x7f, 0x28);
static const IID CLSID_TimeLogServer =
{0xa433e701, 0xe45e, 0x11d3, {0x97, 0xb5, 0x52, 0x54, 0x4c, 0xba, 0x7f, 0x28}};
// 聲明接口GUID{A433E702-E45E-11d3-97B5-52544CBA7F28}
//DEFINE_GUID(IID_ITimeLog,
//0xa433e702, 0xe45e, 0x11d3, 0x97, 0xb5, 0x52, 0x54, 0x4c, 0xba, 0x7f, 0x28);
static const IID IID_ITimeLog =
{0xa433e702, 0xe45e, 0x11d3, {0x97, 0xb5, 0x52, 0x54, 0x4c, 0xba, 0x7f, 0x28}};

  2.寫一個ITimeLogServer.h,在ITimeLogServer.h文件中聲明組件和接口

//ITimeLogServer.h
#include ";GUIDs.h";
//接口ITimeLog的聲明
DECLARE_INTERFACE_(ITimeLog,IUnknown)
{
    STDMETHOD(OutputLog)(BSTR* varLogText)PURE;
};

  說明:

  1.宏DEFINE_GUID將組件和接口的progid與GUID相關聯.可以用guidgen.exe工具產生。

  2.宏DECLARE_INTERFACE_聲明接口;該宏第一個參數為接口名,第二個參數為該接口的基類.聲明沒有基類的接口用DECLARE_INTERFACE宏。

  3.宏STDMETHOD聲明接口中的方法.此方法的返回值為HRESULT.PURE被解釋為";=0";,即此方法為純虛函數.當方法的返回值不是HRESULT時,用宏STDMETHOD_(返回類型,函數名)(參數)PURE;

  二.聲明組件類CTimeLogServer和實現接口的嵌套類

  在ClassWizard中添加新類CTimeLogServer,其基類選擇為CCmdTarget.修改其頭文件TimeLogServer1.h,加上#include ";ITimeLogServer.h";;同時在類聲明體中加上

//聲明實現ITimelog接口的嵌套類
    BEGIN_INTERFACE_PART(TimeLog,ITimeLog)//自動聲明IUnknown接口的三個方法
    STDMETHOD(OutputLog)(BSTR* varLogText);
    END_INTERFACE_PART(TimeLog)
    //聲明接口映射
    DECLARE_INTERFACE_MAP()
    //聲明類廠
    DECLARE_OLECREATE(CTimeLogServer)

  三.實現類廠和接口映射

  在CTimeLogServer的實現文件中寫入:

//實現類廠
IMPLEMENT_OLECREATE(CTimeLogServer,";TimeLogServer";,
0xa433e701, 0xe45e, 0x11d3, 0x97, 0xb5, 0x52, 0x54, 0x4c, 0xba, 0x7f, 0x28);
//映射接口到相應的嵌套類
BEGIN_INTERFACE_MAP(CTimeLogServer,CCmdTarget)
INTERFACE_PART(CTimeLogServer,IID_ITimeLog,TimeLog)
END_INTERFACE_MAP()


  四.在組件的構造和析構函數中對全局對象計數

CTimeLogServer::CTimeLogServer()
{
    ::AfxOleLockApp();
}

CTimeLogServer::~CTimeLogServer()
{
    ::AfxOleUnlockApp();
}
 
  五.為嵌套類實現IUnknown接口

//為嵌套類而實現IUnknown接口
STDMETHODIMP_(ULONG)
CTimeLogServer::XTimeLog::AddRef()
{
    METHOD_PROLOGUE(CTimeLogServer,TimeLog)
    return pThis->;ExternalAddRef();
}

STDMETHODIMP_(ULONG)
CTimeLogServer::XTimeLog::Release()
{
    METHOD_PROLOGUE(CTimeLogServer,TimeLog)
    return pThis->;ExternalRelease();
}

STDMETHODIMP
CTimeLogServer::XTimeLog::QueryInterface(REFIID riid,void**ppvObj)
{
    METHOD_PROLOGUE(CTimeLogServer,TimeLog)
    return pThis->;ExternalQueryInterface(&;riid,ppvObj);
}
 
  說明:雖然CCmdTarget類已經實現了IUnknown接口,但是還必須通過上述代碼來將嵌套類的IUnknown映射到CCmdTarget支持的IUnknown接口.METHOD_PROLOGUEH宏的兩個參數分別是實現組件對象的類和實現接口的嵌套類。

  六.實現ItimeLog接口的方法OutputLog

  注意本組件的功能是往日志文件中輸入日志.

  1. 在組件類中添加一個文件指針:

 // Attributes
 public:
 protected:
      FILE* m_logfile;


  2. 初始化和退出

  首先在CTimeLogServer的構造函數中進行一些初始化:

CTimeLogServer::CTimeLogServer()
{
    ::AfxOleLockApp();
    CTime TimeStamp = CTime::GetCurrentTime();
    CString FileName;
    FileName.Format(_T(";%s.log";),TimeStamp.Format(";%Y%m%d";));
    m_logfile = fopen(FileName,_T(";a";));
    if(m_logfile)
    {
        fprintf(m_logfile,_T(";# # # # # # # # # # # # # # # # # ";));
        fprintf(m_logfile,_T(";開始於:%s";),(LPCTSTR)TimeStamp.Format(";%Y年%m月%d日%H:%M %S";));
        fprintf(m_logfile,_T("; ";));
    }
}
//然後在析構函數中關閉文件
CTimeLogServer::~CTimeLogServer()
{
    ::AfxOleUnlockApp();
    if(m_logfile)
    {
        CTime TimeStamp = CTime::GetCurrentTime();
        fprintf(m_logfile,_T("; ";));
        fprintf(m_logfile,_T(";結束於:%s";),(LPCTSTR)TimeStamp.Format(";%Y年%m月%d日%H:%M %S";));
    fprintf(m_logfile,_T("; ";));
        fprintf(m_logfile,_T(";# # # # # # # # # # # # # # # # # ";));
        fclose(m_logfile);
    }
}

  3. 實現接口ITimeLog方法

//實現接口ITimeLog方法
STDMETHODIMP
CTimeLogServer::XTimeLog::OutputLog(BSTR* varLogText)
{
    METHOD_PROLOGUE(CTimeLogServer,TimeLog)
    if(pThis->;m_logfile)
    {
        CTime TimeStamp = CTime::GetCurrentTime();
        CString NowTime = TimeStamp.Format(";%Y年%m月%d日%H:%M:%S";);
        CString LogText((LPCWSTR)*varLogText);
fprintf(pThis->;m_logfile,"; %s %s %";,NowTime,LogText);
        return NOERROR;
    }
    else
    {
AfxMessageBox(";沒有日志文件!";);
        return S_FALSE;
    }
}


  七.完善組件服務器

  在應用對象CTimeLogServerApp的 實現文件中,處理Instance()和ExitInstance()

BOOL CTimeLogServerApp::InitInstance()
{
    ::AfxOleLockApp();
    // Register all OLE server (factorIEs) as running. This enables the
    // OLE librarIEs to create objects from other applications.
    COleObjectFactory::RegisterAll();

    return TRUE;
}
int CTimeLogServerApp::ExitInstance()
{
    // TODO: Add your specialized code here and/or call the base class
  ::AfxOleUnlockApp();
    return CWinApp::ExitInstance();
}
 


  第二節 客戶程序

  使用COM組件服務器的客戶程序關鍵步驟是:初始化COM庫,創建組件對象並獲取IUnknown接口指針,查詢接口並使用,釋放組件。

  #include ";ITimeLogServer.h";
  //初始化COM庫,對組件實例化
    HRESULT hResult;
  IUnknown* pIUnknown;
    hResult = ::CoInitialize(NULL);
    if(FAILED(hResult))
    {
        ::AfxMessageBox(";不能初始化COM庫!";);
        return FALSE;
    }

  //創建組件實例
  pIUnknown = NULL;
    hResult = ::CoCreateInstance(CLSID_TimeLogServer,NULL,
        CLSCTX_INPROC_SERVER,IID_IUnknown,(void**)&;pIUnknown);
    if(FAILED(hResult))
    {
        pIUnknown = NULL;
        ::AfxMessageBox(";不能創建TimeLog對象!";);
        return FALSE;
    }
  //查詢接口並使用
  if(pIUnknown!=NULL)
        {
            ITimeLog* pITimeLog;
HResult=pIUnknown->;QueryInterface(IID_ITimeLog,(void**)&;pITimeLog);
            if(FAILED(hResult))
            {
            ::AfxMessageBox(";不能獲取接口ITimeLog!";);
                pIUnknown->;Release();
                return;
            }
            BSTR bstrLogText;
            bstrLogText = m_logtext.AllocSysString();
            CString text((LPCWSTR)bstrLogText);
            ::AfxMessageBox(text);

            if(FAILED(pITimeLog->;OutputLog(&;bstrLogText)))
            {
                ::AfxMessageBox(";日志輸出出錯!";);
                pITimeLog->;Release();
                return;
            }
            pITimeLog->;Release();
            ::AfxMessageBox(";日志已經寫入!";);
        }
    //釋放組件
    pIUnknown->;Release();
    pIUnknown = NULL;
      ::CoUninitialize();

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved