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

<2009年4月>
2930311234
567891011
12131415161718
19202122232425
262728293012
3456789

統(tǒng)計(jì)

  • 隨筆 - 24
  • 文章 - 0
  • 評(píng)論 - 17
  • 引用 - 0

常用鏈接

留言簿(4)

隨筆分類(lèi)

隨筆檔案

相冊(cè)

搜索

  •  

最新評(píng)論

閱讀排行榜

評(píng)論排行榜

用C實(shí)現(xiàn)的一個(gè)基本COM接口IFoo(來(lái)自COM Programmer's Cookbook)——C實(shí)現(xiàn)COM接口系列1

用C實(shí)現(xiàn)的一個(gè)基本COM接口IFoo(來(lái)自COM Programmer's Cookbook)


把該文中實(shí)現(xiàn)的代碼整理匯總到一個(gè)項(xiàng)目中。目前只是實(shí)現(xiàn)到一個(gè)中間階段,重點(diǎn)在說(shuō)明COM接口的實(shí)現(xiàn)原理,還沒(méi)有包含類(lèi)廠的部分。以后還需陸續(xù)添加類(lèi)廠等高級(jí)功能。

文件組成:
ifoo.h         COM接口IFoo,接口ID IID_IFoo 聲明文件。
outside.c      COM接口實(shí)現(xiàn)。這里實(shí)現(xiàn)IFoo的是一個(gè)結(jié)構(gòu)體COutside.
util.h         一些宏定義、全局函數(shù)、變量聲明文件。
main.c         筆者為實(shí)現(xiàn)項(xiàng)目添加的文件。提供main函數(shù)、內(nèi)存管理函數(shù)Alloc,Free的實(shí)現(xiàn)(封裝C運(yùn)行庫(kù)函數(shù)malloc和free.)、接口ID定義。

COM接口到底是什么?
COM接口是一個(gè)指向虛函數(shù)表的指針。通過(guò)這個(gè)指針可以訪問(wèn)內(nèi)存中某處的各個(gè)功能塊,執(zhí)行預(yù)定義的功能,完成用戶的任務(wù)。這些功能塊以函數(shù)的形式存在(想不出還有其他形式:))并被調(diào)用。它們有一個(gè)共同點(diǎn):都包含一個(gè)指針參數(shù),指向這些功能要操作的數(shù)據(jù)地址。在C++中,這個(gè)地址就是對(duì)象的首地址,也就是類(lèi)成員函數(shù)中隱含的this指針。在C函數(shù)中并沒(méi)有這種現(xiàn)成的便利,因此代碼實(shí)現(xiàn)中在接口定義時(shí)仍使用了接口指針(HRESULT (__stdcall * QueryInterface)   (IFoo * This,  const IID * const, void **)),而在接口函數(shù)實(shí)現(xiàn)時(shí)根據(jù)結(jié)構(gòu)體布局結(jié)構(gòu),從這個(gè)接口指針推算得到對(duì)象實(shí)例指針。

typedef struct IFoo
{
 struct IFooVtbl * lpVtbl;
} IFoo;
typedef struct IFooVtbl IFooVtbl;
struct IFooVtbl
{
 
 HRESULT (__stdcall * QueryInterface)   (IFoo * This,  const IID * const, void **) ;
 ULONG (__stdcall * AddRef)    (IFoo * This) ;
 ULONG (__stdcall * Release)   (IFoo * This) ;

 HRESULT (__stdcall * SetValue)         (IFoo * This,  int) ;
 HRESULT (__stdcall * GetValue)         (IFoo * This,  int *) ;
};

COM接口的要求:

每一個(gè)COM接口(指向的虛函數(shù)表)的頭三個(gè)函數(shù)必須是IUnknown接口的函數(shù):QueryInterface,AddRef和Release。在C++中,稱為從IUnknown接口繼承。
對(duì)于調(diào)用QueryInterface響應(yīng)查詢IID_IUnknwon得到的接口指針值,同一個(gè)對(duì)象實(shí)現(xiàn)的所有接口必須相同。這是判斷兩個(gè)COM對(duì)象是否是同一個(gè)對(duì)象的標(biāo)準(zhǔn)。

 

宏定義“#define IUNK_VTABLE_OF(x) ((IUnknownVtbl *)((x)->lpVtbl))“說(shuō)明

在預(yù)處理輸出文件main.i中可以找到IUnknownVtbl和IFooVtbl的聲明:
    typedef struct IUnknownVtbl
    {
       
       
        HRESULT ( __stdcall *QueryInterface )(
            IUnknown * This,
             const IID * const riid,
             void **ppvObject);
       
        ULONG ( __stdcall *AddRef )(
            IUnknown * This);
       
        ULONG ( __stdcall *Release )(
            IUnknown * This);
       
       
    } IUnknownVtbl;
   
    struct IUnknown
    {
         struct IUnknownVtbl *lpVtbl;
    };   
   

struct IFooVtbl
{
 
 HRESULT (__stdcall * QueryInterface)   (IFoo * This,  const IID * const, void **) ;
 ULONG (__stdcall * AddRef)    (IFoo * This) ;
 ULONG (__stdcall * Release)   (IFoo * This) ;

 HRESULT (__stdcall * SetValue)         (IFoo * This,  int) ;
 HRESULT (__stdcall * GetValue)         (IFoo * This,  int *) ;
};

該宏定義的作用就是把IFoo接口中的IFooVtbl類(lèi)型的指針拿出來(lái)((x)->lpVtbl)),并強(qiáng)制轉(zhuǎn)換((IUnknownVtbl *))成IUnknownVtbl。
“強(qiáng)制轉(zhuǎn)換”的結(jié)果是什么呢?是怎么做到的呢?
很明顯,結(jié)果就是得到的指針不再是IFooVtbl *類(lèi)型,而是變成了IUnknownVtbl *類(lèi)型。至于做法,系統(tǒng)應(yīng)該記錄每一個(gè)變量、表達(dá)式的類(lèi)型。當(dāng)進(jìn)行強(qiáng)制類(lèi)型轉(zhuǎn)換時(shí),就(臨時(shí)地)修改其類(lèi)型為轉(zhuǎn)換到的類(lèi)型。
同理,QueryInterface, AddRef, Release宏定義中的(IUnknown *)也是這種用法。

可以看到,宏“IUNK_VTABLE_OF“的作用是供宏QueryInterface,宏AddRef,宏Release引用,把IFooVtbl *類(lèi)型轉(zhuǎn)換為IUnknownVtbl *類(lèi)型,最終達(dá)到調(diào)用IUnknownVtbl中定義的三個(gè)QueryInterface,AddRef,Release函數(shù)。

那么,這種大費(fèi)周章的目的是什么呢?為什么不以IFooVtbl中三個(gè)函數(shù)的定義形式(不通過(guò)強(qiáng)制轉(zhuǎn)換來(lái)轉(zhuǎn)換成必須的類(lèi)型),直接調(diào)用IFooVtbl中定義的函數(shù)呢?雖然強(qiáng)制轉(zhuǎn)換在參數(shù)值上并不會(huì)造成改變,最終調(diào)用的也是IFooVtbl定義的函數(shù)(FooQueryInterface,FooAddRef,FooRelease)。

為什么一定要通過(guò)IUnknown接口指針調(diào)用這三個(gè)函數(shù)呢?修改QueryInterface宏定義如下:
#define QueryInterface(pif, iid, pintf) \
 (((pif)->lpVtbl)->QueryInterface(pif, iid, (void **)(pintf)))
即通過(guò)IFoo接口指針來(lái)調(diào)用由IUnknown引入的函數(shù),有什么不對(duì)的地方嗎?

試驗(yàn)表明,將QueryInterface宏定義如下也可以編譯通過(guò),執(zhí)行起來(lái)也沒(méi)有出現(xiàn)任何異常。
#define QueryInterface(pif, iid, pintf) \
 (((pif)->lpVtbl)->QueryInterface(pif, iid, (void **)(pintf)))

 

對(duì)于IUnknown接口的三個(gè)函數(shù),調(diào)用時(shí)傳遞的參數(shù)是IUnknown *類(lèi)型(見(jiàn)QueryInterface, AddRef, Release宏定義),而函數(shù)定義中(FooQueryInterface, FooAddRef, FooRelease)聲明的參數(shù)是IFoo *類(lèi)型,這種不一致的情況是怎么出現(xiàn)的?這種不一致不會(huì)有問(wèn)題嗎?

這種不一致的產(chǎn)生是由于從不同的角度看待引起的。如果從IUnknown接口來(lái)看,那么接口函數(shù)中的第一個(gè)參數(shù)類(lèi)型就是IUnknown *;如果從IFoo來(lái)看,那么第一個(gè)參數(shù)的類(lèi)型就是IFoo *。

這種不一致性只是針對(duì)于編譯器對(duì)于類(lèi)型的編譯要求有意義的,在接口實(shí)現(xiàn)及使用時(shí),傳遞給lpVtbl->QueryInterface, lpVtbl->AddRef,lpVtbl->Release的第一個(gè)參數(shù)在值上都是相同的,都是實(shí)現(xiàn)該接口的內(nèi)存地址(在本例中是COutside對(duì)象的首地址)。

 

一些語(yǔ)法現(xiàn)象回顧

函數(shù)指針變量定義、賦值及調(diào)用。
HRESULT (__stdcall * pQI)   (IFoo * This,  const IID * const, void **) ;
定義一個(gè)函數(shù)指針變量pQI,該變量指向“返回HRESULT,取3個(gè)參數(shù)分別為類(lèi)型IFoo *,const IID * const, void **”的函數(shù)。

typedef HRESULT (__stdcall * QIType)   (IFoo * This,  const IID * const, void **) ;
定義一個(gè)函數(shù)指針類(lèi)型,該類(lèi)型的指針指向“返回HRESULT,取3個(gè)參數(shù)分別為類(lèi)型IFoo *,const IID * const, void **”的函數(shù)。

HRESULT __stdcall QueryInterface(IFoo * This,  const IID * const, void **) ;//函數(shù)聲明示例
pQI = 0;   // 函數(shù)指針賦值,0表示不指向任何函數(shù)。
pQI = QueryInterface;  // 函數(shù)指針賦值,pQI指向QueryInterface。
pQI = &QueryInterface; // 與上面等價(jià)。

QueryInterface(&this->ifoo, riid, ppv);  // 使用函數(shù)名直接調(diào)用
pQI(&this->ifoo, riid, ppv);             // 函數(shù)指針調(diào)用
(*pQI)(&this->ifoo, riid, ppv);          // 第二種函數(shù)指針調(diào)用方式


宏定義、展開(kāi)規(guī)則
對(duì)于宏,一直有一種霧里看花的感覺(jué),似乎很隨意,怎么來(lái)都行,比如:
#define AddRef(pif) \
 (IUNK_VTABLE_OF(pif)->AddRef((IUnknown *)(pif)))

宏定義應(yīng)該是可以嵌套的,即宏定義的“內(nèi)容“中還可以包含(嵌套)宏,如本例,“IUNK_VTABLE_OF”就是嵌套宏。在展開(kāi)的時(shí)候,將嵌套的宏也一并展開(kāi)(替換成定義的內(nèi)容),直到不再有宏為止。
那么就有兩個(gè)疑問(wèn):
1。如果被嵌套的宏包含(直接或間接)定義的宏,那么展開(kāi)就沒(méi)完沒(méi)了,死循環(huán)了。
2。如果定義的內(nèi)容中有跟定義的宏同名的字符串(比如上面的例子IUNK_VTABLE_OF),那么怎么區(qū)分這同名的東東是嵌套的宏(需要展開(kāi)),還是一般的字符串(不需要展開(kāi))?


函數(shù)調(diào)用規(guī)范約定、main函數(shù)調(diào)用規(guī)范。

一開(kāi)始把幾個(gè)文件匯總到項(xiàng)目里時(shí),編譯通不過(guò),錯(cuò)誤提示大致意思是,不能把一種調(diào)用規(guī)范的函數(shù)指針轉(zhuǎn)換成另一種調(diào)用規(guī)范的函數(shù)指針。后來(lái)把調(diào)用規(guī)范改為   /Gz(__stdcall),編譯為(Compile As)改為/TC(Compile As C Code)就好了。

想來(lái)是對(duì)于.c文件,編譯器缺省使用的是__cdecl,而IFoo中的接口宏定義在win32下展開(kāi)成了__stdcall,所以出現(xiàn)了矛盾。而使用/Gz強(qiáng)制未聲明調(diào)用規(guī)范的函數(shù)使用__stdcall,實(shí)現(xiàn)就與聲明一致了。


(size_t)&(((s *)0)->m)

c++程序員也許都知道,訪問(wèn)地址“0”處的成員是一大忌,會(huì)造成GP。然而,取地址“0”處的成員的地址,卻是個(gè)合法的操作。雖然地址“0”處并沒(méi)有什么內(nèi)容,但是,如果在地址0處存放一個(gè)內(nèi)容,那么該內(nèi)容中的成員也是有地址的。本例中正是巧妙地利用這種方法,從接口地址計(jì)算得出實(shí)現(xiàn)該接口的實(shí)例地址,進(jìn)而訪問(wèn)實(shí)例的內(nèi)部變量。

------------------------------------------------------------------------------------
2009年5月6日
附上源碼:/Files/gracelee/outside.zip

代碼執(zhí)行結(jié)果:

posted on 2009-04-29 12:08 小蔥蘸醬 閱讀(2007) 評(píng)論(3)  編輯 收藏 引用 所屬分類(lèi): 編碼點(diǎn)滴

評(píng)論

# re: 用C實(shí)現(xiàn)的一個(gè)基本COM接口IFoo(來(lái)自COM Programmer's Cookbook)——C實(shí)現(xiàn)COM接口系列1 2009-04-29 21:40 likenk

很好,現(xiàn)在講com的資料非常少,希望lz能繼續(xù)下去,講的易懂些。先謝謝啦。

# re: 用C實(shí)現(xiàn)的一個(gè)基本COM接口IFoo(來(lái)自COM Programmer's Cookbook)——C實(shí)現(xiàn)COM接口系列1 2009-04-30 11:16 xcpp

強(qiáng)大

# re: 用C實(shí)現(xiàn)的一個(gè)基本COM接口IFoo(來(lái)自COM Programmer's Cookbook)——C實(shí)現(xiàn)COM接口系列1 2009-05-17 11:03 flyceit

真是太棒了!希望作者堅(jiān)持下去.
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美国产日韩a欧美在线观看| 另类人畜视频在线| 99精品国产99久久久久久福利| 美女啪啪无遮挡免费久久网站| 国产亚洲欧美另类一区二区三区| 先锋影音网一区二区| 亚洲一区二区三区四区视频| 国产精品日日摸夜夜添夜夜av| 亚洲综合好骚| 亚洲综合精品自拍| 国产亚洲制服色| 美女精品自拍一二三四| 久久综合伊人77777麻豆| 亚洲黄色性网站| 亚洲国内自拍| 牛人盗摄一区二区三区视频| 亚洲美女视频网| 99re这里只有精品6| 欧美日韩亚洲网| 小黄鸭精品密入口导航| 欧美一区亚洲二区| 一区二区在线观看视频| 欧美黄色成人网| 欧美日韩一卡| 欧美专区在线播放| 久久综合导航| 一本色道**综合亚洲精品蜜桃冫| 在线视频你懂得一区二区三区| 国产精品自拍网站| 免费成人黄色av| 欧美日韩国产123| 亚洲欧美久久久| 久久久精品午夜少妇| 亚洲日韩成人| 中文国产成人精品| 激情文学综合丁香| 亚洲精品久久久久久久久| 国产精品视频你懂的| 老司机67194精品线观看| 欧美激情小视频| 欧美一区二区三区四区在线观看地址| 久久久99爱| 一区二区三区视频观看| 欧美一区二区三区在线播放| 91久久精品久久国产性色也91| 一本色道久久综合亚洲精品婷婷| 国产日韩欧美不卡| 亚洲国产精品一区二区尤物区| 国产精品久久久久一区二区| 免费亚洲一区二区| 欧美色区777第一页| 久久午夜视频| 欧美日韩成人综合| 久久资源在线| 欧美午夜精品久久久久久人妖| 久久这里有精品15一区二区三区| 欧美精品一区在线发布| 久久九九久精品国产免费直播| 欧美黄色免费| 久久久综合网站| 欧美日韩在线观看一区二区| 美日韩精品视频免费看| 国产精品久久久久9999| 欧美激情91| 国产日韩欧美电影在线观看| 亚洲精品乱码久久久久久久久 | 久久精品国产精品亚洲| 夜夜精品视频| 久久美女性网| 欧美一区二区三区免费观看| 欧美高潮视频| 老鸭窝亚洲一区二区三区| 国产精品二区影院| 亚洲国产精品欧美一二99| 国产揄拍国内精品对白| 一本久久青青| 亚洲麻豆国产自偷在线| 久久精品午夜| 欧美在线视频观看| 欧美日韩综合网| 亚洲国产精品久久久久久女王| 国产亚洲美州欧州综合国| 亚洲午夜小视频| 99热免费精品在线观看| 乱中年女人伦av一区二区| 久久久成人网| 欧美与黑人午夜性猛交久久久| 国产欧美日韩精品一区| 亚洲人成精品久久久久| 亚洲高清不卡在线| 欧美在线免费看| 性色一区二区三区| 欧美日韩一卡| 亚洲精品一区二区三区四区高清| 亚洲第一页自拍| 久久精品夜色噜噜亚洲a∨| 欧美一区二区在线播放| 欧美性猛交xxxx乱大交退制版| 最新国产の精品合集bt伙计| 亚洲国产欧美一区二区三区同亚洲| 午夜视频精品| 欧美一区二区三区四区夜夜大片| 欧美四级剧情无删版影片| 亚洲精品专区| 一本一道久久综合狠狠老精东影业| 免费高清在线视频一区·| 麻豆精品精华液| 一区二区三区亚洲| 久久国产精品99精品国产| 久久av一区二区三区| 国产伦精品一区二区| 一区二区三区精密机械公司| 在线视频中文亚洲| 欧美日韩国产成人高清视频| 亚洲破处大片| 一区二区三区四区五区视频 | 亚洲美女一区| 欧美精品成人91久久久久久久| 亚洲国产精品123| 亚洲日本va午夜在线影院| 欧美jizz19hd性欧美| 亚洲第一中文字幕在线观看| 亚洲欧洲在线一区| 欧美激情中文不卡| 亚洲精品在线免费| 亚洲一区在线免费观看| 国产精品国产精品国产专区不蜜| 一区二区三区欧美成人| 亚洲欧美资源在线| 国产伦理一区| 久久精品30| 欧美成人一区二区三区在线观看| 在线观看91精品国产麻豆| 蜜桃久久av一区| 亚洲人成网站影音先锋播放| 一区二区三区 在线观看视| 欧美性色aⅴ视频一区日韩精品| 在线亚洲免费视频| 欧美一区二区在线免费观看| 国内精品久久久久久久影视麻豆 | 宅男精品视频| 国产精品色网| 欧美在线播放视频| 欧美福利电影在线观看| 99av国产精品欲麻豆| 欧美午夜视频网站| 午夜激情综合网| 牛夜精品久久久久久久99黑人 | 欧美日韩国产一区二区三区| 亚洲视频免费在线| 久久久久99| 亚洲精品123区| 欧美午夜不卡在线观看免费| 午夜性色一区二区三区免费视频 | 欧美国产日韩精品| 亚洲一区二区三区在线视频| 久久久一区二区| 亚洲人www| 国产精品日韩在线一区| 久久嫩草精品久久久精品| 91久久精品国产| 亚洲欧洲av一区二区三区久久| 国产亚洲成av人片在线观看桃| 久久免费视频观看| 99精品久久久| 久久综合狠狠| 一区二区三区日韩在线观看| 国产一二三精品| 欧美激情综合亚洲一二区| 亚洲男女毛片无遮挡| 欧美成人亚洲| 午夜欧美不卡精品aaaaa| 在线看视频不卡| 欧美午夜精品一区| 久久久免费观看视频| 99国产一区| 米奇777在线欧美播放| 亚洲一区bb| 在线成人亚洲| 国产精品久久一级| 免费看av成人| 先锋影音一区二区三区| 亚洲国产综合91精品麻豆| 久久精品成人欧美大片古装| 99re66热这里只有精品3直播| 国产欧美二区| 欧美日韩国产在线观看| 久久精品国产第一区二区三区最新章节| 亚洲精品自在在线观看| 久久五月天婷婷| 亚洲免费伊人电影在线观看av| 亚洲高清成人| 国产欧美一区二区三区视频| 欧美伦理91i| 久久手机免费观看| 亚洲永久免费| 日韩视频精品在线观看| 欧美成人免费一级人片100| 欧美一区二区三区四区视频| 一本一本久久|