国产精品一区二区精品_久久小视频_天堂va在线观看_99久久夜色精品国产亚洲96_日本手机在线视频_av成人免费

當(dāng)前位置:首頁 > 網(wǎng)站舊欄目 > 學(xué)習(xí)園地 > 設(shè)計軟件教程 > 深入分析D語言接口與COM接口的關(guān)系

深入分析D語言接口與COM接口的關(guān)系
2010-01-13 21:04:58  作者:佚名  來源:


深入分析D語言接口與COM接口的關(guān)系
   
    前兩天為了解決dxpcom項目中遇到的xpcom接口兼容性問題,看了一下DMD編譯器的源碼,對D的接口有了一些了解,現(xiàn)在總結(jié)出來,備忘。

    D中有了專門用于標(biāo)識接口的關(guān)鍵字interface,而不用象C++中使用抽象類來代替。
    D代碼:
Java代碼
interface ITest  
{  
int test();  

interface ITest
{
int test();
}
    C++代碼:
Java代碼
class ITest  
{  
int test()=0;  

class ITest
{
int test()=0;
}

    而D中的接口與C++中的接口不同之處是,D中的接口仍然含有ClassInfo,存放在虛表的0項上。

    從DMD的源碼中可以得知,D中的類,接口都在虛表的0項上保存了ClassInfo指針。
    這樣,D中的接口是無法與C++接口兼容的,則D就無法調(diào)用Windows的COM對象,至少是無法“優(yōu)雅”的調(diào)用(仍然可以使用struct進(jìn)行二進(jìn)制兼容代替)。

    為了解決這個問題,DMD就需要能夠表示出與C++兼容的COM接口,即需要一個虛表是"干凈"的接口。又由于,從一個COM接口繼承的接口仍然是一個COM接口,而COM模型的實現(xiàn)上又恰好定義了一個“IUnknown”根接口(COM體系中的所有的接口都是繼承了IUnknown)。

    所以,出于簡單實現(xiàn)的原則,DMD區(qū)分一個接口是D接口還是COM接口,關(guān)鍵就是判斷這個接口是不是叫做IUnknown,以及這個接口是否繼承自IUnknown,雖然接口都是通過Interface關(guān)鍵字聲明。更有趣的是,DMD僅僅判斷接口的名字是否為"IUnknown"而根本不管接口中的方法如何定義。

    以上所述內(nèi)容在進(jìn)行Windows COM編程時,幾乎不會被察覺,因為Windows的所有接口都是繼承自IUnknown,只要正常使用就可以了。

    而在進(jìn)行Mozilla xpcom編程的時候,xpcom的根接口叫做ISupports,DMD根本就不會認(rèn)為這是需要編譯為C++兼容的COM接口,而仍然會將虛表的0項進(jìn)行保留,結(jié)果給使用者造成了虛表指針偏移了的印象。

    基于D的這個識別COM接口的方式,在dxpcom項目中,qiezi使用了別名的方式進(jìn)行了變換,既將dxpcom項目中的所有的接口名稱進(jìn)行了優(yōu)雅的統(tǒng)一,又能夠使DMD生成正確的COM接口:
Java代碼
extern(Windows)  
interface IUnknown {  
  static const char[] IID_STR = NS_ISUPPORTS_IID_STR;  
  static const nsIID IID = NS_ISUPPORTS_IID;  
 
  /* void QueryInterface (in nsIIDRef uuid, [iid_is (uuid), retval] out nsQIResult result); */ 
  nsresult QueryInterface(nsIID * uuid, void * *result);  
 
  /* [noscript, notxpcom] nsrefcnt AddRef (); */ 
  nsrefcnt AddRef();  
 
  /* [noscript, notxpcom] nsrefcnt Release (); */ 
  nsrefcnt Release();  
 
}  
 
alias IUnknown nsISupports; 

extern(Windows)
interface IUnknown {
  static const char[] IID_STR = NS_ISUPPORTS_IID_STR;
  static const nsIID IID = NS_ISUPPORTS_IID;

  /* void QueryInterface (in nsIIDRef uuid, [iid_is (uuid), retval] out nsQIResult result); */
  nsresult QueryInterface(nsIID * uuid, void * *result);

  /* [noscript, notxpcom] nsrefcnt AddRef (); */
  nsrefcnt AddRef();

  /* [noscript, notxpcom] nsrefcnt Release (); */
  nsrefcnt Release();

}

alias IUnknown nsISupports;

   這個現(xiàn)象同時也很好的說明了,D中的別名(alias)在符號的處理方面僅僅是一個符號的替換,同C/C++中的#define的作用相同。

    下面的兩段代碼就能很好的詮釋本文的內(nèi)容(感謝qiezi提供)

    代碼一,無法通過運(yùn)行期斷言,因為接口IInterface仍然為標(biāo)準(zhǔn)D接口,虛表的0項為ClassInfo指針無法被顯示的調(diào)用,在執(zhí)行的結(jié)果中就表現(xiàn)為虛表進(jìn)行了偏移。
Java代碼
extern(Windows):     
int test1(IInterface p)     
{     
    return 1;     
}     
    
int test2(IInterface p)     
{     
    return 2;     
}     
    
int test3(IInterface p)     
{     
    return 3;     
}     
    
struct InterfaceVtbl     
{     
extern(Windows):     
    int function(IInterface) test1;     
    int function(IInterface) test2;     
    int function(IInterface) test3;     
}     
    
struct Interface     
{     
    InterfaceVtbl* vtbl;     
    
    InterfaceVtbl vtbl_;     
    
    static Interface opCall()     
    {     
        Interface res;     
        res.vtbl_.test1 = &test1;     
        res.vtbl_.test2 = &test2;     
        res.vtbl_.test3 = &test3;     
        res.vtbl = &res.vtbl_;     
        return res;     
    }     
}     
    
interface IInterface     
{     
    int test1();     
    int test2();     
    int test3();     
}     
    
extern (D):     
    
void main()     
{     
    Interface i = Interface();     
    assert(i.vtbl.test1(cast(IInterface)&i) == 1);     
    assert(i.vtbl.test2(cast(IInterface)&i) == 2);     
    assert(i.vtbl.test3(cast(IInterface)&i) == 3);     
    
    IInterface ii = cast(IInterface)&i;     
    assert(ii.test1() == 1);     
    assert(ii.test2() == 2);     
    assert(ii.test3() == 3);     

extern(Windows):  
int test1(IInterface p)  
{  
    return 1;  
}  
 
int test2(IInterface p)  
{  
    return 2;  
}  
 
int test3(IInterface p)  
{  
    return 3;  
}  
 
struct InterfaceVtbl  
{  
extern(Windows):  
    int function(IInterface) test1;  
    int function(IInterface) test2;  
    int function(IInterface) test3;  
}  
 
struct Interface  
{  
    InterfaceVtbl* vtbl;  
 
    InterfaceVtbl vtbl_;  
 
    static Interface opCall()  
    {  
        Interface res;  
        res.vtbl_.test1 = &test1;  
        res.vtbl_.test2 = &test2;  
        res.vtbl_.test3 = &test3;  
        res.vtbl = &res.vtbl_;  
        return res;  
    }  
}  
 
interface IInterface  
{  
    int test1();  
    int test2();  
    int test3();  
}  
 
extern (D):  
 
void main()  
{  
    Interface i = Interface();  
    assert(i.vtbl.test1(cast(IInterface)&i) == 1);  
    assert(i.vtbl.test2(cast(IInterface)&i) == 2);  
    assert(i.vtbl.test3(cast(IInterface)&i) == 3);  
 
    IInterface ii = cast(IInterface)&i;  
    assert(ii.test1() == 1);  
    assert(ii.test2() == 2);  
    assert(ii.test3() == 3);  
}

    代碼二,與代碼一的結(jié)構(gòu)完全一致,卻能夠通過運(yùn)行時斷言的檢查。唯一的不同僅僅是IInterface的名字換成了IUnknown!!
Java代碼
extern(Windows):     
int test1(IUnknown p)     
{     
    return 1;     
}     
    
int test2(IUnknown p)     
{     
    return 2;     
}     
    
int test3(IUnknown p)     
{     
    return 3;     
}     
    
struct InterfaceVtbl     
{     
extern(Windows):     
    int function(IUnknown) test1;     
    int function(IUnknown) test2;     
    int function(IUnknown) test3;     
}     
    
struct Interface     
{     
    InterfaceVtbl* vtbl;     
    
    InterfaceVtbl vtbl_;     
    
    static Interface opCall()     
    {     
        Interface res;     
        res.vtbl_.test1 = &test1;     
        res.vtbl_.test2 = &test2;     
        res.vtbl_.test3 = &test3;     
        res.vtbl = &res.vtbl_;     
        return res;     
    }     
}     
    
interface IUnknown     
{     
    int test1();     
    int test2();     
    int test3();     
}     
    
extern (D):     
    
void main()     
{     
    Interface i = Interface();     
    assert(i.vtbl.test1(cast(IUnknown)&i) == 1);     
    assert(i.vtbl.test2(cast(IUnknown)&i) == 2);     
    assert(i.vtbl.test3(cast(IUnknown)&i) == 3);     
    
    IUnknown ii = cast(IUnknown)&i;     
    assert(ii.test1() == 1);     
    assert(ii.test2() == 2);     
    assert(ii.test3() == 3);     

extern(Windows):  
int test1(IUnknown p)  
{  
    return 1;  
}  
 
int test2(IUnknown p)  
{  
    return 2;  
}  
 
int test3(IUnknown p)  
{  
    return 3;  
}  
 
struct InterfaceVtbl  
{  
extern(Windows):  
    int function(IUnknown) test1;  
    int function(IUnknown) test2;  
    int function(IUnknown) test3;  
}  
 
struct Interface  
{  
    InterfaceVtbl* vtbl;  
 
    InterfaceVtbl vtbl_;  
 
    static Interface opCall()  
    {  
        Interface res;  
        res.vtbl_.test1 = &test1;  
        res.vtbl_.test2 = &test2;  
        res.vtbl_.test3 = &test3;  
        res.vtbl = &res.vtbl_;  
        return res;  
    }  
}  
 
interface IUnknown  
{  
    int test1();  
    int test2();  
    int test3();  
}  
 
extern (D):  
 
void main()  
{  
    Interface i = Interface();  
    assert(i.vtbl.test1(cast(IUnknown)&i) == 1);  
    assert(i.vtbl.test2(cast(IUnknown)&i) == 2);  
    assert(i.vtbl.test3(cast(IUnknown)&i) == 3);  
 
    IUnknown ii = cast(IUnknown)&i;  
    assert(ii.test1() == 1);  
    assert(ii.test2() == 2);  
    assert(ii.test3() == 3);  
}

    另外需要說明的是extern(D),extern(Windows),extern(Pascal)等特征,只是用來描述函數(shù)的調(diào)用約定,與接口的類型無關(guān)。
    一句話:D中的類與標(biāo)準(zhǔn)D接口都有ClassInfo在虛表的0項上,而COM接口的虛表是干凈的;而將一個接口聲明為COM接口的方式為:將這個接口命名為IUnknown或繼承自IUnknown。

 
更多信息請登陸http://61.191.27.74:802/ 最后,歡迎加入http://61.191.27.74:802/的會員


安徽新華電腦學(xué)校專業(yè)職業(yè)規(guī)劃師為你提供更多幫助【在線咨詢
国产精品一区二区精品_久久小视频_天堂va在线观看_99久久夜色精品国产亚洲96_日本手机在线视频_av成人免费
<button id="0mgmq"><pre id="0mgmq"></pre></button>
  • <tr id="0mgmq"></tr>
  • <abbr id="0mgmq"><source id="0mgmq"></source></abbr> <button id="0mgmq"></button>
  • 国产区日韩欧美| 欧美日韩亚洲免费| 国产精品二区三区四区| 久久久久欧美| 免费一区视频| 999在线观看精品免费不卡网站| 日韩精品久久久毛片一区二区| 不卡视频一区二区| 亚洲三级影院| 欧美视频网站| 午夜视频久久久| 亚洲国产精品日韩| 西游记1978| 欧美高清视频一区二区三区在线观看| 久久三级视频| 91青青草免费观看| 亚洲综合国产| 麻豆成人精品| 久久久蜜桃一区二区人| 99精品国产福利在线观看免费| 欧美三级小说| 国内精品国语自产拍在线观看| 欧美黄污视频| 精品白丝av| 亚洲精品少妇| 国产精品久久久久久久免费软件 | 欧美激情aⅴ一区二区三区| 奇米精品在线| 婷婷精品国产一区二区三区日韩| 玛丽玛丽电影原版免费观看1977| 国产综合av一区二区三区| 国产麻豆乱码精品一区二区三区| 动漫精品视频| 蜜桃导航-精品导航| 日韩av图片| 欧美国产精品| 99亚洲精品| 97在线中文字幕| 欧美1o一11sex性hdhd| 欧洲一区二区在线| 午夜精品999| 亚洲视频成人| 国产在线精品一区二区三区》| 国产精品免费观看高清| 日本不卡高清视频一区| 亚洲综合网中心| 在线播放不卡| 成人免费视频网站入口| 久久99精品久久久久子伦| 亚洲乱码一区二区三区| 欧美激情日韩| 久久久久久国产精品mv| 久久精品成人一区二区三区蜜臀| 亚洲国产激情一区二区三区| 最新成人av网站| 国产精品乱子乱xxxx| 亚洲激情啪啪| 亚洲在线一区| 日本亚洲欧洲精品| 在线欧美视频| 不卡视频一区| 午夜久久久久| 国产精品视频免费一区| 午夜国产精品视频| 成人精品一二区| 在线观看精品视频| 久久不射中文字幕| 亚洲成人蜜桃| 久久久久se| 一区二区免费电影| 91精品国产一区二区三区动漫| 午夜欧美性电影| 麻豆九一精品爱看视频在线观看免费| 欧美精品免费观看二区| 一本色道久久综合亚洲精品不| 欧美日韩另类丝袜其他| 99热精品在线| 一区二区不卡在线观看| 久久免费黄色| 在线观看欧美亚洲| 久久精品国产一区二区三区日韩| 亚洲人体一区| 欧美在线亚洲| 欧美激情一区二区三区在线视频 | 色99中文字幕| 老**午夜毛片一区二区三区| 欧美国产免费| 视频一区视频二区视频三区高| 久久精品国产清高在天天线| 欧美精品不卡| 日本精品一区二区三区高清 久久| 亚洲在线播放| 日韩视频一区| 在线不卡视频一区二区| 久久99精品久久久久久三级| 国产亚洲欧美一区二区| 你懂的亚洲视频| 日韩欧美亚洲日产国| 国产欧美亚洲日本| 久久国产66| 国产农村妇女毛片精品久久莱园子 | 伊人狠狠色j香婷婷综合| 免费国产一区| 精品国产二区在线| 久久综合亚州| 国产精品外国| aa亚洲婷婷| 亚洲国产欧美国产综合一区| 欧美大片一区| 中文字幕一区二区中文字幕| 欧美日韩亚洲一区二区三区四区| 国产精品日韩一区二区免费视频| 亚洲免费在线| 午夜一区在线| 久久国产精品久久久久久电车| 国产日韩欧美一区在线| 国内一区二区在线视频观看| 一区二区三区四区欧美| 亚洲高清视频一区| 日本午夜精品一区二区| 久久久久久久久久码影片| 国产超碰91| 精品国产福利| 视频一区二区在线观看| 水蜜桃亚洲精品| 亚洲精品成人三区| 欧美精选在线| 激情综合视频| 国产毛片一区| 国产经典一区二区三区| 国产精选在线观看91| 国产综合欧美在线看| 久久综合伊人77777麻豆| 欧美三级网色| 曰韩不卡视频| 亚洲视频大全| 翡翠波斯猫1977年美国| 欧美高清性xxxxhdvideosex| 欧美影视一区二区| 中文精品一区二区三区| 亚洲无线一线二线三线区别av| 9久re热视频在线精品| 99c视频在线| 国产一区二区三区免费不卡| 日本不卡久久| 好看的日韩av电影| **亚洲第一综合导航网站 | 日本视频一区在线观看| 一区二区视频在线播放| 亚洲激情亚洲| 国产一区二区黄色| 一区二区精品在线观看| 国产人成精品一区二区三| 国产精品毛片va一区二区三区| 亚洲 日韩 国产第一区| 中文国产一区| 久久久精品动漫| 好吊视频一区二区三区四区| 99在线影院| 一区高清视频| 久久av一区二区| 日韩在线导航| 免费一区视频| 一区二区三区国| 久久国产毛片| 日韩影视精品| 久久亚洲综合| 伊人久久婷婷色综合98网| 国产欧美一级| 亚洲一区二区三区免费观看| 亚洲美女一区| 日本不卡二区| 久久久国产精品一区二区中文| 亚洲成人第一| 国产精品日韩一区二区三区| 欧美精品一区二区视频 | 亚洲欧美日韩一区在线观看| 日韩激情视频| caoporn国产精品免费公开| 欧美一区二区视频在线| 99久久久精品免费观看国产| 国产精品magnet| 欧美日本国产精品| 亚洲欧美高清| 国产精品黄色| 日韩激情久久| 国产精品久久久久久免费观看| 欧美午夜精品久久久久免费视| 久99久在线| 3d动漫啪啪精品一区二区免费 | 亚洲看片网站| 国产尤物99| 久久激情中文| 亚洲毛片在线| 欧美日韩在线高清| 欧洲亚洲一区二区三区四区五区| 91九色蝌蚪嫩草| 99人久久精品视频最新地址| 在线看成人av电影| 日韩av一区二区三区在线观看|