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

huaxiazhihuo

 

C++的非侵入式接口

終于寫到c++的非侵入式接口了,興奮,開心,失望,解脫,…… 。在搞了這么多的面向?qū)ο罂破罩螅救艘惨呀?jīng)開始不耐煩,至此,不想做太多闡述。

雖然,很早就清楚怎么在c++下搞非侵入式接口,但是,整個(gè)框架代碼,重構(gòu)了十幾次之后,才終于滿意。支持給基本類型添加接口,好比intcharconst char*double;支持泛型,好比vectorlist;支持繼承,基類實(shí)現(xiàn)的接口,表示子類也繼承了對(duì)該接口的實(shí)現(xiàn),而且子類也可以拒絕基類的接口,好比鴨子拒絕基類鳥類“會(huì)飛”,編譯時(shí)報(bào)錯(cuò);支持接口組合;……,但是,這里僅僅簡(jiǎn)單介紹其原理,并不涉及C++中各種變態(tài)細(xì)節(jié)的處理,C++中,但凡是要正兒八經(jīng)的稍微做點(diǎn)正事,就要面臨無窮無盡的細(xì)節(jié)糾結(jié)。

先看看其使用例子:

1、自然是定義一個(gè)接口:取之于真實(shí)代碼片段

    struct IFormatble
    {
        static TypeInfo* GetTypeInfo();
       
virtual void Format(TextWriter& stream, const FormatInfo& info) = 0;
        
virtual bool Parse(TextReader& stream, const FormatInfo& info)
        {
            PPNotImplement();
        }
    };

2、接口的實(shí)現(xiàn)類,假設(shè)為int添加IFormatble的接口實(shí)現(xiàn),實(shí)際代碼肯定不會(huì)這樣對(duì)一個(gè)一個(gè)的基本類型來寫實(shí)現(xiàn)類的代碼。這里只是為了舉例說明。類的名字就隨便起好啦,

    struct ImpIntIFormatble : IFormatble
    {
        
int* mThis;    //這一行是關(guān)鍵
        virtual void Format(TextWriter& stream, const FormatInfo& info)override
        {}

        
virtual bool Parse(TextReader& stream, const FormatInfo& info)override
        {}
    };

這里的關(guān)鍵是,實(shí)現(xiàn)類的字段被規(guī)定死了,最多只能包含3個(gè)指針成員字段,且第1個(gè)字段一定是目的類型指針,第二是類型信息對(duì)象(用于泛型),第三是額外參數(shù),次序不能亂。成員字段如果不需要用到第二第三個(gè)成員字段數(shù)據(jù),可以省略不寫,好比這里。所有接口實(shí)現(xiàn)類必須遵守這樣的內(nèi)存布局;

3、裝配,將接口的實(shí)現(xiàn)類裝配到現(xiàn)有的類上,以告訴編譯器該類對(duì)于某個(gè)接口(這里為IFormatble)的實(shí)現(xiàn),用的是第2步的實(shí)現(xiàn)類ImpIntIFormatble

PPInterfaceOf(IFormatble, int, ImpIntIFormatble);

 

4、將實(shí)現(xiàn)類注冊(cè)到類型信息的接口實(shí)現(xiàn)列表中,這一步可以省略,只是為了運(yùn)行時(shí)的接口查詢,相當(dāng)于IUnknownQuery。這一行代碼是在全局對(duì)象的構(gòu)造函數(shù)中執(zhí)行的,放在cpp源文件中

RegisterInterfaceImp<IFormatble, int>();
然后就可以開開心心地使用接口了,比如
            int aa = 20;
            TextWriter stream();
            FormatInfo info();
            TInterface
<IFormatble> formatable(aa); //TInterface這個(gè)名字過難看,也沒辦法了
            formatable
->Format(stream, info);
            
double dd = 3.14;
            formatable 
= TInterface<IFormatble>(dd);    //假設(shè)double也實(shí)現(xiàn)IFormatble
            formatable->Format(stream, info);

是否有點(diǎn)神奇呢?其實(shí)也沒什么,不過就是在trait和內(nèi)存布局上做文章,也就只是用了類型運(yùn)算的伎倆。考察ImpIntIFormatble的內(nèi)存布局,對(duì)于普遍的c++編譯器來說,對(duì)象的虛函數(shù)表指針(如果存在的話),都放在對(duì)象的起始地址上,后面緊跟對(duì)象本身的成員數(shù)據(jù)字段,因此,ImpIntIFormatble的內(nèi)存布局相當(dāng)于,

struct ImpIntIFormatble
{
    
void* vtbl;
    
int* mThis;
};

 

注意,這里已經(jīng)沒有繼承了。這就是,實(shí)現(xiàn)了IFormatble 接口的ImpIntIFormatble對(duì)象的內(nèi)存表示。因此,可以想象,所有的接口實(shí)現(xiàn)類的內(nèi)存布局都強(qiáng)制規(guī)定為以下形式:

    struct InterfaceLayout
    {
        
const void* mVtbl;
        
const void* mThis;            //對(duì)象本身
        const TypeInfo* mTypeInfo;    //類型信息
        const void* mParam;    //補(bǔ)充參數(shù),一般很少用到
    };

當(dāng)然,如果編譯器的虛函數(shù)表指針不放在對(duì)象起始地址的話,就沒法這么玩了,那么非侵入式接口也無從做起。然后,就是TInterface了,繼承于InterfaceLayout

    template<typename IT>
    
struct TInterface : public InterfaceLayout
    {
        typedef IT interface_type;
        static_assert(is_abstract
<IT>::value, "interface must have pure function");
        static_assert(
sizeof(IT) == sizeof(void*), "Can't have data");
    
public:
        interface_type
* operator->()const
        {
            interface_type
* result = (interface_type*)(void*)this;
            
return result;
        }
        
    };

不管怎么說都好,TInterface對(duì)象的內(nèi)存布局與接口實(shí)現(xiàn)類的內(nèi)存布局一致。因此操作符->重載函數(shù)才可以粗暴的類型轉(zhuǎn)換來順利完成。然后構(gòu)造TInterface對(duì)象的時(shí)候就是強(qiáng)制獲取ImpIntIFormatble對(duì)象的虛函數(shù)表(也就是其起始地址的指針數(shù)據(jù))指針賦值給InterfaceLayoutmVtbl,進(jìn)而依次把實(shí)際對(duì)象的指針放在mThis上,獲取到類型信息對(duì)象放在mTypeInfo中,如果有必要搭理mParam,也相應(yīng)地賦值。

然后,就是template<typename Interface, typename Object>struct InterfaceOf各種特化的運(yùn)用而已,就不值一提了。

由于c++abi沒有統(tǒng)一標(biāo)準(zhǔn),并且,c++標(biāo)準(zhǔn)也沒有規(guī)定編譯器必須用虛函數(shù)表來實(shí)現(xiàn)多態(tài),所以,這里的奇技淫巧并不能保證在所有平臺(tái)上都能夠成立,但是,非侵入式接口真是方便,已經(jīng)是本座寫c++代碼的核心工具,一切都圍繞著非侵入式接口來展開。

原本打算長(zhǎng)篇大論,也只有草草收?qǐng)觥V螅咀徒夥帕耍瑫?huì)暫時(shí)離開cppblog很久,計(jì)劃中的內(nèi)容,消息發(fā)送,虛模板函數(shù),字符串,輸入輸出,格式化,序列化, locale,全局變量,模板表達(dá)式,組合子解析器,allocator,智能指針,程序運(yùn)行時(shí),抽象工廠訪問者等模式的另類實(shí)現(xiàn),以求從全新的角度上來表現(xiàn)C++的強(qiáng)大,也只能中斷了。








posted on 2017-07-15 17:01 華夏之火 閱讀(2941) 評(píng)論(2)  編輯 收藏 引用 所屬分類: c++技術(shù)探討

評(píng)論

# re: C++的非侵入式接口 2017-07-17 11:03 天下

cppblog人氣不旺啊,
csdn還不錯(cuò),
現(xiàn)在都懶的寫blog了,唉.
  回復(fù)  更多評(píng)論   

# re: C++的非侵入式接口 2017-07-28 09:43 萬連文

確實(shí)沒必要在普及這些知識(shí)了,時(shí)代變了。雖然道理不會(huì)變,但是誰有在意這些呢?  回復(fù)  更多評(píng)論   

導(dǎo)航

統(tǒng)計(jì)

常用鏈接

留言簿(6)

隨筆分類

隨筆檔案

搜索

積分與排名

最新評(píng)論

閱讀排行榜

評(píng)論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            久久久五月天| 在线播放一区| 亚洲淫性视频| 亚洲一区二区在| 亚洲专区一区二区三区| 亚洲欧美一区二区激情| 欧美一区二区日韩| 久久久91精品国产| 欧美激情日韩| 国产精品一区二区三区久久久| 国产精品一二三四区| 狠狠色综合网站久久久久久久| 激情伊人五月天久久综合| 亚洲电影免费| 这里只有精品视频| 欧美国产亚洲精品久久久8v| 久久久噜噜噜久久中文字免| 蜜桃av久久久亚洲精品| 亚洲第一主播视频| 亚洲人成久久| 亚洲欧美日韩精品久久久久| 久久婷婷影院| 国产精品第十页| 激情久久五月| 亚洲欧美一区二区三区极速播放 | 亚洲免费在线视频一区 二区| 欧美亚洲免费| 亚洲黄色在线| 欧美在线资源| 国产精品夫妻自拍| 亚洲青色在线| 久久久久久亚洲综合影院红桃| 亚洲欧洲日产国产综合网| 午夜国产一区| 欧美日韩少妇| 亚洲黄色三级| 玖玖国产精品视频| 午夜精品一区二区三区在线| 免费观看成人鲁鲁鲁鲁鲁视频 | 久久免费精品视频| 国产精品永久免费视频| 亚洲毛片在线免费观看| 久热精品在线视频| 欧美亚洲在线| 国产精品一区二区久久| 在线亚洲观看| 亚洲精品美女久久久久| 免费不卡在线观看av| 韩国一区二区三区在线观看| 亚洲欧美视频在线观看视频| 亚洲人在线视频| 欧美激情精品久久久久久免费印度| 国内揄拍国内精品少妇国语| 久久国产精品久久国产精品| 亚洲一区二区视频在线| 欧美日韩中文在线观看| 一区二区免费在线视频| 亚洲日本激情| 欧美日韩国产片| 在线视频日韩| 一区二区三区四区国产| 欧美视频在线观看免费网址| 99精品视频免费观看| 亚洲精品一区二| 欧美日韩国产专区| 中国成人黄色视屏| 亚洲性av在线| 精品粉嫩aⅴ一区二区三区四区| 久久深夜福利免费观看| 亚洲黄色免费电影| 亚洲国产精品va在线看黑人 | 亚洲国产精品黑人久久久| 欧美高清在线视频观看不卡| 免费一级欧美片在线观看| 亚洲国产精品va在线看黑人| 亚洲欧洲另类国产综合| 国产精品观看| 久久久久久久网站| 久久亚洲视频| 一区二区三区精品| 亚洲午夜激情网站| 在线观看成人一级片| 亚洲高清在线| 国产精品呻吟| 麻豆av一区二区三区久久| 牛人盗摄一区二区三区视频| 一区二区成人精品| 欧美一区二区视频在线| 亚洲精品午夜精品| 香蕉av福利精品导航| 亚洲国产婷婷香蕉久久久久久99| 91久久精品国产91久久性色tv| 国产精品大全| 欧美成人一区二区三区在线观看| 欧美va天堂在线| 欧美一区二区三区喷汁尤物| 牛人盗摄一区二区三区视频| 亚洲欧美日韩国产综合| 久久免费99精品久久久久久| 一区二区三区免费在线观看| 性做久久久久久久免费看| 亚洲精品久久久久| 欧美亚洲综合在线| 一区二区三区欧美成人| 久久久91精品国产一区二区三区| 99国产精品久久久| 欧美在线一级视频| 亚洲欧美日韩第一区| 欧美国产免费| 蜜臀久久久99精品久久久久久| 国产精品黄色在线观看| 亚洲国产国产亚洲一二三| 国产精品日韩欧美一区| 亚洲黄色在线视频| 精品动漫3d一区二区三区免费版 | 欧美阿v一级看视频| 国产精品网红福利| 日韩视频一区二区三区在线播放| 在线观看一区| 欧美在线免费观看视频| 午夜精品久久久久久| 欧美精品一区二区视频| 欧美高清你懂得| 亚洲福利视频网站| 久久久久在线观看| 久久爱www久久做| 国产精品激情| 日韩视频一区二区| 亚洲视频一区在线| 久久综合国产精品| 欧美成人精品h版在线观看| 黑人操亚洲美女惩罚| 宅男66日本亚洲欧美视频| 亚洲乱码一区二区| 欧美成人有码| 亚洲国产精品久久人人爱蜜臀| 影音先锋国产精品| 久久国产精品99国产精| 久久久青草青青国产亚洲免观| 国产精品系列在线播放| 亚洲综合导航| 久久精品久久99精品久久| 国产区在线观看成人精品| 欧美一级专区| 免费一级欧美片在线观看| 亚洲国产成人av好男人在线观看| 久久成人免费| 亚洲大胆人体在线| 亚洲精品国久久99热| 欧美伦理影院| 亚洲网站在线看| 久久久久久婷| 91久久精品网| 欧美日韩精品免费看| 亚洲视频免费在线| 久久久精品国产免大香伊| 一区视频在线看| 欧美精品在欧美一区二区少妇| 一本一本久久a久久精品综合麻豆 一本一本久久a久久精品牛牛影视 | 国产老女人精品毛片久久| 欧美一区二区三区视频免费播放| 久久久精品国产99久久精品芒果| 亚洲国产91精品在线观看| 欧美理论片在线观看| 亚洲一区精品视频| 久久综合给合| 亚洲性人人天天夜夜摸| 国产一区欧美| 欧美伦理视频网站| 午夜精品久久久久久久久久久| 美女国内精品自产拍在线播放| 亚洲黄一区二区三区| 国产精品a级| 久久久亚洲成人| 亚洲午夜女主播在线直播| 蜜臀久久99精品久久久久久9| 亚洲视频每日更新| 精品69视频一区二区三区| 欧美日本一区二区三区 | 欧美影院在线| 亚洲欧洲日产国产综合网| 久久激情一区| 亚洲视频免费| 亚洲激情偷拍| 国产亚洲一区二区三区在线播放| 欧美精品日韩一区| 久久亚洲春色中文字幕| 亚洲综合色网站| 日韩一级大片在线| 亚洲第一区色| 免费在线成人| 久久久久国产一区二区| 国产精品国产精品| 欧美成人一区二区三区| 久久国产精品99国产精| 亚洲女优在线| 亚洲免费影视| 亚洲第一精品夜夜躁人人躁| 久久全国免费视频| 亚洲国产精品一区二区www在线|