• <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>
            <2007年6月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            1234567

            統計

            • 隨筆 - 24
            • 文章 - 0
            • 評論 - 17
            • 引用 - 0

            常用鏈接

            留言簿(4)

            隨筆分類

            隨筆檔案

            相冊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            用C實現的一個基本COM接口IFoo(來自COM Programmer's Cookbook)——C實現COM接口系列2

            C實現COM接口系列1中實現的com接口IFoo與使用它的客戶耦合在一起,沒有實現在各自分離的模塊,因此不符合模塊化編程思想。本期添加類廠支持,以使接口的實現與接口的使用相分離。

            ---------------------------------------------------
            類廠的作用到底是什么?
            將接口的實現與客戶使用分離開來嗎?

            不盡然。使用CoCreateInstance,客戶可以完全不必知道類廠的存在,而創建組件,獲取組件實現的接口并使用。

            即COM庫可以完全拋開類廠的概念,而是提供一個這樣的函數原型:
            CoCreateObject(REFID rclsid,...,REFID riid,void **ppItf);
            用戶在調用的時候可以對riid提供IID_Unknown或者特定于該對象的一個接口,直接獲取該對象的IUnknown或特定的接口指針。

            可以看到,這正是CoCreateInstance所作的事情。

            1 類廠提供了間接創建類對象的方式:用戶可以先獲取并持有類廠接口指針,通過該指針所指向的類廠接口創建類對象。適用于需要創建多個(或重復創建)類對象的地方,減少了每次都要定位對象庫并把對象庫裝入內存的開銷。
            2 類廠提供了保證組件庫留在內存不被卸載出去的另一種方法:類廠接口函數LockServer。組件庫維護一個庫范圍計數器,只有該計數器為0時,組件庫才允許自己被卸載出內存。(與此相對,引用計數是類對象范圍的,通過該類實現的各個接口來維護。如果一個類對象的引用計數達到0,那么該對象占有的內存就被釋放,該對象上的接口指針也不再有效。)
            除了調用LockServer鎖定組件庫以外,當創建的組件個數大于0時,組件庫也不能被卸載。也可以說,調用一次LockServer()的作用相當于創建了一個組件。

            -----------------------------------------------------------------------
            客戶一側:
            1 使用一個接口需要知道哪些信息?
            備選:
            接口IID
            類對象(類廠)CLSID(或ProgID)
            接口函數原型(參數個數,類型,返回值)
            實現接口組件的線程模型(進程內、進程外、遠程)?
            類型庫typelib信息?

            服務一側:
            2 實現一個組件和接口以供客戶調用,需要提供哪些東西?
            備選:
            所有客戶使用組件和接口所需的內容
            額外的還有:


            --------------------------------------------------------------------
            為dll添加.def文件與直接在需要導出的函數定義處指定_declspec( dllexport )有區別嗎?如果有是什么區別?

            我發現在outdll.c中這樣指定:
            __declspec( dllexport ) HRESULT DllGetClassObject (REFCLSID rclsid, REFIID riid, void **ppv)
            會產生編譯錯誤:
            1>------ Build started: Project: outside, Configuration: Debug Win32 ------
            1>Compiling...
            1>outdll.c
            1>d:\outside-cf\outside\outdll.c(19) : error C2375: 'DllGetClassObject' : redefinition; different linkage
            1>        c:\program files\microsoft visual studio 8\vc\platformsdk\include\objbase.h(833) : see declaration of 'DllGetClassObject'
            1>Build log was saved at "file://d:\outside-cf\outside\Debug\BuildLog.htm"
            1>outside - 1 error(s), 0 warning(s)
            ========== Build: 0 succeeded, 1 failed, 1 up-to-date, 0 skipped ==========

            c2375的解釋意思是出錯的函數使用的鏈接指示符與之前聲明的不同。
            Compiler Error C2375
            'function' : redefinition; different linkage

            The function is already declared with a different linkage specifier.

            objbase.h中聲明了DllGetClassObject()函數:
            STDAPI  DllGetClassObject(IN REFCLSID rclsid, IN REFIID riid, OUT LPVOID FAR* ppv);

            而使用.def文件就沒有問題。

            -----------------------------------------------------------------------------
            初次執行結果:

            問題就是總有一個分配的內存沒有釋放:

             


            根據打印出來的內存地址可以判斷,應該是先創建的類廠對象的內存沒有釋放。
            檢查代碼,main()中并沒有忘記調用Release(pCF)釋放類廠對象。打印Release(pCF)的返回值,發現是1,即在類廠接口指針上少調用了一次Release,那么,究竟是哪里少的呢?

            main()函數中有關類廠對象引用計數的地方就是CoGetClassObject和Release(CreateInstance跟類廠自己的引用計數無關),這是一對增加引用計數和減少引用計數的對應操作,所以,main()中應該沒有問題。

            那么,就只有創建類廠對象的時候了。下面看一下類廠對象是如何創建的。
            首先,main調用CoGetClassObject,該函數就調用dll中的DllGetClassObject。由于是第一次調用(不考慮其他客戶使用該dll的情況),程序執行到CreateClassFactory(...),該函數執行完后,類廠對象的引用計數是1。

            由于創建成功,因此繼續向下執行到QueryInterface,此時,類廠對象的引用計數變成了2。然后,DllGetClassObject返回,com庫函數CoGetClassObject也應該返回。注意,此時的類廠對象引用計數已經是2了!

            因此,問題就出在這里。main調用一次CoGetClassObject后,類廠對象的引用計數是2,而不是我想向中的1。于是,后面調用一次Release也就當然無法釋放掉類場對象了。

             

             

             1 HRESULT DllGetClassObject (REFCLSID rclsid, REFIID riid, void **ppv)
             2 {
             3     *ppv = 0;
             4     if (IsEqualCLSID (rclsid, &CLSID_Outside))
             5     {
             6 
             7         if (!vpcfOutside)
             8 
             9         {
            10 
            11             HRESULT hr = CreateClassFactory (&CLSID_Outside, CreateOutside,
            12                                              &IID_IClassFactory, &vpcfOutside);
            13 
            14                 if (hr != NOERROR)
            15 
            16                     return hr;
            17         }
            18 
            19         return QueryInterface (vpcfOutside, riid, ppv);
            20 
            21     }
            22 
            23     return E_FAIL;
            24 }

             

             


            找到了原因,改正就很容易了。這里我覺得需要把DllGetClassObject作如下修改:

             1 HRESULT DllGetClassObject (REFCLSID rclsid, REFIID riid, void **ppv)
             2 {
             3     *ppv = 0;
             4     if (IsEqualCLSID (rclsid, &CLSID_Outside))
             5     {
             6 
             7         if (!vpcfOutside)
             8 
             9         {
            10 
            11             HRESULT hr = CreateClassFactory (&CLSID_Outside, CreateOutside,
            12                                              &IID_IClassFactory, &vpcfOutside);
            13 
            14                 if (hr != NOERROR)
            15 
            16                     return hr;
            17 
            18     if(IsEqualIID(riid,&IID_IClassFactory))
            19     {
            20      *ppv = vpcfOutside;// Set *ppv to vpcfOutside directly instead of QueryInterface if first time creation
            21      return NOERROR;
            22     }
            23     else
            24     {
            25      Release(vpcfOutside);// Any interface requested (riid) other than IID_ClassFactory and IID_Unknown not support by class factory,
            26                           // call Release to free the memory.
            27      return E_FAIL;
            28     }
            29 
            30         }
            31 
            32         return QueryInterface (vpcfOutside, riid, ppv);
            33 
            34     }
            35 
            36     return E_FAIL;
            37 }

             

            修改后在執行,內存都正常釋放了。

            -------------------------------------------------------------------------------------------
            CreateClassFactory代碼說明

             1 HRESULT CreateClassFactory (REFCLSID rclsid,
             2     HRESULT (*pfnCreate)(IUnknown *, REFIID, void **), 
             3     REFIID riid, void **ppv)
             4 {
             5     ClassFactory *this;
             6     HRESULT hr;
             7     
             8     *ppv = 0;
             9     if (hr = Alloc (sizeof (ClassFactory), &this))
            10     return hr;
            11 
            12     this->icf.lpVtbl = &vtblClassFactory;
            13     this->cRef = 1;  // After this call, cRef==1
            14 
            15     this->pfnCreate = pfnCreate;
            16     
            17     hr = QueryInterface (&this->icf, riid, ppv);  // After this call, cRef==2
            18     Release (&this->icf);  // Corresponds to "this->cRef = 1", ater this call, cRef==1
            19 
            20     return hr;
            21 }

             

            可以看到,兩行代碼的效果是對引用計數增1及減1,這兩行代碼執行后,對引用計數的影響互相抵消,等于沒有改變引用計數。那么,把這兩行同時注釋掉,是不是可以呢?
            我的回答是:在本例中可以。因為這兩行代碼之間的QueryInterface總是可以執行成功的(因為是用IDD_ClassFactory來調用該函數的)。所以,即便把這兩行代碼同時注釋掉,CreateClassFactory執行結束后,類廠對象的引用計數也增了1,以后調用Release就可以釋放掉類廠對象占用的內存。
            但是,如果CFQueryInterface的代碼編寫中除了錯誤,比如,像這樣寫:

             1 static HRESULT CFQueryInterface (IClassFactory *pcf, REFIID riid, void **ppv)
             2 {
             3     ClassFactory *this = IMPL (ClassFactory, icf, pcf);
             4 
             5     if (IsEqualIID (riid, &IID_IUnknown) ||
             6 //            IsEqualIID (riid, &IID_IClassFactory))   // Comment out this condition to create an error
             7         *ppv = &this->icf;
             8     else
             9     {
            10         *ppv = 0;
            11         return E_NOINTERFACE;
            12     }
            13 
            14     AddRef ((IClassFactory *)*ppv);
            15 
            16     return NOERROR;
            17 }


            那么,這兩行代碼之間的QueryInterface就會執行出錯,那么類廠對象占用的內存就永遠沒有機會釋放了。
            也就是說,AddRef和Release雖然在作用上對引用計數來說相互抵消,但Release函數提供了釋放對象內存的機會(當引用計數為0時),如果不成對的調用他們,也就失去了管理對象內存(釋放對象占用的內存)的機會。

            ---------------------------------------------------------------------------
            組件庫outside文件說明:
             IFoo.h      IFoo接口聲明
             outside.c   組件對象、IFoo接口實現
             cf.c        類廠對象、IClassFactory接口實現
             outdll.c    組件庫導出函數實現
             outside.def 組件庫模塊定義文件,導出函數聲明
             outside.reg 組件庫注冊文件

            ----------------------------------------------------------------------------
            源碼: outside-cf

            posted on 2009-05-11 16:28 小蔥蘸醬 閱讀(1145) 評論(0)  編輯 收藏 引用 所屬分類: 編碼點滴

            亚洲精品WWW久久久久久| 亚洲国产精品无码久久久蜜芽| 久久久久久无码Av成人影院| 久久Av无码精品人妻系列| 国内精品久久久久| 性高朝久久久久久久久久| 久久久老熟女一区二区三区| 国产精品亚洲综合专区片高清久久久| 久久伊人五月天论坛| 777米奇久久最新地址| 性做久久久久久免费观看| 国产69精品久久久久777| 东方aⅴ免费观看久久av | 免费精品国产日韩热久久| 久久亚洲美女精品国产精品| 久久久国产精华液| 久久久久四虎国产精品| 伊人久久大香线蕉综合热线| 久久97久久97精品免视看秋霞| 久久er99热精品一区二区| 久久中文字幕人妻丝袜| 久久人妻少妇嫩草AV蜜桃| 青青青国产精品国产精品久久久久 | 国产婷婷成人久久Av免费高清| 一本大道久久东京热无码AV| 91精品婷婷国产综合久久| 国产情侣久久久久aⅴ免费| 精品国产99久久久久久麻豆| 亚洲精品成人久久久| 无码任你躁久久久久久久| 国内精品伊人久久久久影院对白| 久久久无码精品亚洲日韩按摩| 久久精品国产99国产精品导航 | 久久天天躁狠狠躁夜夜2020一| 办公室久久精品| 色综合色天天久久婷婷基地| 久久精品国产精品青草app| 国产精品久久久久久久| 99久久精品国产麻豆| 亚洲国产精品婷婷久久| 99久久精品免费看国产一区二区三区|