• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            笑看風云淡

            寵辱不驚,看庭前花開花落;去留無意,望天空云卷云舒
            posts - 96, comments - 48, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 ::  :: 聚合  :: 管理
            MFC中是通過嵌套類而不是多重繼承來實現COM接口的,通過接口映射機制將接口和實現該接口的嵌套類關聯起來;MFC中提供一套簡明的宏來實現嵌套類的定義

            首先應當明確,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(";# # # # # # # # # # # # # # # # # \n";));
                    fprintf(m_logfile,_T(";開始于:%s";),(LPCTSTR)TimeStamp.Format(";%Y年%m月%d日%H:%M %S";));
                    fprintf(m_logfile,_T(";\n";));
                }
            }
            //然后在析構函數中關閉文件
            CTimeLogServer::~CTimeLogServer()
            {
                ::AfxOleUnlockApp();
                if(m_logfile)
                {
                    CTime TimeStamp = CTime::GetCurrentTime();
                    fprintf(m_logfile,_T(";\n";));
                    fprintf(m_logfile,_T(";結束于:%s";),(LPCTSTR)TimeStamp.Format(";%Y年%m月%d日%H:%M %S";));
                fprintf(m_logfile,_T(";\n";));
                    fprintf(m_logfile,_T(";# # # # # # # # # # # # # # # # #\n";));
                    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,";\n%s\n%s\n%";,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();
            久久AV高清无码| 国产∨亚洲V天堂无码久久久| 好属妞这里只有精品久久| 国产精品久久久久影院嫩草| 2021久久国自产拍精品| 久久国产综合精品五月天| 久久午夜福利电影| 久久精品国产2020| 91久久成人免费| 久久精品国产亚洲av麻豆图片| 亚洲av日韩精品久久久久久a| 国产真实乱对白精彩久久| 国产99久久久国产精品小说| 久久久精品国产sm调教网站| 精品一久久香蕉国产线看播放| 无码伊人66久久大杳蕉网站谷歌 | 伊人 久久 精品| 久久精品国产99久久久古代| 久久综合久久美利坚合众国| 亚洲精品高清国产一久久| 伊人久久大香线蕉亚洲| 久久99精品久久久久久9蜜桃| 国产高潮国产高潮久久久| 亚洲AV伊人久久青青草原| 91精品婷婷国产综合久久| 国产国产成人精品久久| 囯产精品久久久久久久久蜜桃 | 性欧美丰满熟妇XXXX性久久久 | 97精品伊人久久久大香线蕉| 狼狼综合久久久久综合网| 中文字幕无码久久精品青草| 一本大道加勒比久久综合| AV狠狠色丁香婷婷综合久久| 影音先锋女人AV鲁色资源网久久| 色婷婷狠狠久久综合五月| 久久夜色精品国产亚洲| 久久r热这里有精品视频| 嫩草影院久久国产精品| 伊人色综合久久天天| 久久久久这里只有精品| 日韩欧美亚洲综合久久影院Ds|