青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

<2025年11月>
2627282930311
2345678
9101112131415
16171819202122
23242526272829
30123456

統計

  • 隨筆 - 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 小蔥蘸醬 閱讀(1166) 評論(0)  編輯 收藏 引用 所屬分類: 編碼點滴

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产一区二区日韩| 很黄很黄激情成人| 日韩午夜视频在线观看| 欧美国产第二页| 欧美mv日韩mv亚洲| 一本色道久久99精品综合| 亚洲精品视频在线观看免费| 欧美日韩成人一区二区三区| 亚洲天堂av在线免费观看| 亚洲色图制服丝袜| 国产无一区二区| 欧美成人在线影院| 欧美美女操人视频| 欧美一区二区在线看| 久久米奇亚洲| 在线中文字幕一区| 欧美一区二区三区四区高清| 国内精品伊人久久久久av影院| 欧美成人精品影院| 欧美日韩在线精品| 久久嫩草精品久久久精品| 久久这里有精品视频| 亚洲主播在线播放| 久久综合久久综合久久| 亚洲一卡二卡三卡四卡五卡| 久久av老司机精品网站导航| 日韩视频不卡| 久久精彩视频| 亚洲一区二区三区在线看| 欧美一区二区视频网站| 99精品国产一区二区青青牛奶| 亚洲专区在线| 一区二区免费在线播放| 久久精品视频一| 亚洲永久网站| 欧美极品aⅴ影院| 久久女同精品一区二区| 欧美午夜欧美| 亚洲第一级黄色片| 国产日产欧美精品| 99riav久久精品riav| 精品99一区二区| 香蕉久久一区二区不卡无毒影院| 亚洲美女在线看| 久久精品电影| 欧美一区二区三区喷汁尤物| 欧美精品在线一区| 欧美成人精品| 尤物精品在线| 久久国内精品视频| 欧美在线电影| 国产精品久久九九| 99国产精品久久久| 正在播放欧美视频| 欧美激情网站在线观看| 免费中文日韩| 黄色成人在线网站| 久久国产免费看| 久久手机免费观看| 国内不卡一区二区三区| 欧美一区二区三区另类| 欧美在线免费看| 国产日韩欧美一区| 欧美亚洲免费高清在线观看| 欧美亚洲一区| 国产日韩在线一区二区三区| 亚洲欧美久久久| 久久久久久日产精品| 国产一区二区欧美| 久久国内精品视频| 欧美成人亚洲| 亚洲精品系列| 欧美三级日本三级少妇99| 99精品福利视频| 午夜在线成人av| 国产亚洲精品综合一区91| 欧美影院成年免费版| 老色鬼精品视频在线观看播放| 国内精品久久久久影院 日本资源| 欧美在线播放一区二区| 久久婷婷国产综合精品青草| 一区二区在线观看av| 欧美xxxx在线观看| 99热精品在线观看| 久久成人免费视频| 伊人久久成人| 欧美日韩国产综合一区二区| 亚洲少妇自拍| 另类尿喷潮videofree | 亚洲无线观看| 新片速递亚洲合集欧美合集| 国内精品久久久久国产盗摄免费观看完整版| 亚洲欧美999| 欧美成人自拍| 亚洲男同1069视频| 伊大人香蕉综合8在线视| 欧美 日韩 国产 一区| 一区二区三区日韩精品视频| 久久久水蜜桃av免费网站| 亚洲片国产一区一级在线观看| 国产精品黄页免费高清在线观看| 欧美一级夜夜爽| 亚洲国产欧美一区二区三区久久| 亚洲视频香蕉人妖| 伊人精品久久久久7777| 欧美日韩一区二区三区在线观看免| 欧美在线一二三四区| 亚洲欧洲美洲综合色网| 久久久www成人免费无遮挡大片| 亚洲日本电影| 国产亚洲成精品久久| 欧美日韩色婷婷| 久久躁狠狠躁夜夜爽| 亚洲一区二区三区在线视频| 亚洲国产精品一区二区第一页 | 国产一区二区三区自拍| 欧美日韩精品免费 | 亚洲激情成人网| 久久精品国产在热久久| 一区二区三区久久精品| 精品成人一区二区| 国产精品久久久久久久久久妞妞| 欧美69wwwcom| 久久午夜精品一区二区| 欧美中文在线视频| 午夜精品在线观看| 亚洲视频在线视频| 一本久久a久久精品亚洲| 亚洲大胆美女视频| 免费亚洲电影| 久久综合伊人77777蜜臀| 欧美一级午夜免费电影| 亚洲在线国产日韩欧美| 一本久久精品一区二区| 91久久午夜| 亚洲区免费影片| 亚洲精品视频免费在线观看| 在线日韩视频| 亚洲电影免费观看高清完整版| 韩国v欧美v日本v亚洲v| 国产一区久久| 精品福利av| 在线日韩中文字幕| 在线日韩欧美视频| 亚洲欧洲精品一区二区| 亚洲欧洲在线视频| 99精品国产一区二区青青牛奶 | 好吊一区二区三区| 狠狠色狠狠色综合日日小说| 国产亚洲激情视频在线| 国产一区二区无遮挡| 狠狠色狠狠色综合日日tαg| 一区免费观看视频| 亚洲精品国产无天堂网2021| 日韩一级片网址| 亚洲一区在线观看视频| 香蕉久久国产| 久久视频这里只有精品| 欧美 日韩 国产在线 | 久久丁香综合五月国产三级网站| 久久国产88| 免费一级欧美片在线播放| 亚洲高清激情| 这里只有视频精品| 欧美亚洲综合在线| 免费在线观看日韩欧美| 欧美视频观看一区| 国产一区二区三区日韩| 91久久夜色精品国产网站| 亚洲视频axxx| 久久久久国产精品一区二区| 亚洲电影免费| 亚洲免费一在线| 久久亚洲色图| 国产精品欧美久久| 精品成人在线视频| 亚洲午夜在线视频| 两个人的视频www国产精品| 亚洲精品国产精品国自产在线| 中日韩美女免费视频网站在线观看| 欧美一区三区二区在线观看| 欧美波霸影院| 国产午夜精品在线| 一区二区久久| 蜜桃视频一区| 亚洲一二三区视频在线观看| 久久一区二区三区国产精品| 国产精品久久久爽爽爽麻豆色哟哟| 揄拍成人国产精品视频| 亚洲夜间福利| 亚洲国产成人一区| 久久av一区| 国产精品一区二区久久久久| 99国产精品99久久久久久| 久久综合成人精品亚洲另类欧美| 99热精品在线| 欧美日韩免费观看一区| 亚洲国产精品久久91精品| 久久久久久噜噜噜久久久精品| 亚洲少妇最新在线视频|