• <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>
            隨筆-90  評論-947  文章-0  trackbacks-0

            我們先來看一組接口定義:

             

            struct IX

            {

                virtual void MethodX() = 0;

            };

             

            struct IXA : public IX

            {

                virtual void MethodXA() = 0;

            };

             

            struct IXB : public IX

            {

                virtual void MethodXB() = 0;

            };

             

            struct IXAB : public IXA, public IXB

            {

                virtual void MethodXAB() = 0;

            };

             

            然后我們來實現 IX

             

            struct CX : public IX

            {

                virtual void MethodX()

                {

               

                }

            };

             

            然后再來實現 IXA

             

            struct CXA : public IXA

            {

                virtual void MethodX()

                {

               

                }

             

                virtual void MethodXA()

                {

               

                }

            };

             

            這樣寫當然是沒問題的。但是,有木有發現,這里出現MethodX了。如果,我們需要之前CX中對MethodX的實現,那怎么辦呢?

             

            一開始我傻傻地想,繼承CX不就好了么:

             

            struct CXA : public IXA, public CX

            {

                virtual void MethodXA()

                {

               

                }

            };

             

            結果實例化CXA的時候報錯:

            error C2259: 'CXA' : cannot instantiate abstract class

            due to following members:

            void IX::MethodX(void)' : is abstract

            see declaration of 'IX::MethodX'

             

            (嘿嘿。我不知道是不是挺多人一開始都會這么想的。)究其原因,是因為添加多繼承不會產生覆蓋虛函數的效果。此時虛函數的狀態是:

             

            MethodX

            來自于IXA

            來自于IX

            未實現

            MethodXA

            來自于IXA

             

            CXA實現

            MethodX

            來自于CX

            來自于IX

            CX實現

             

            新添加的CX中的MethodX,只是覆蓋了CX多繼承的IX中的那個MethodX。退一步,即便CX是個獨立的類,沒有繼承于IXCX中的MethoxX也無法為覆蓋IXA中的MethodX。只有處于同一繼承鏈中的各個類才會產生覆蓋虛函數的效果。

             

            假如,CX就是繼承于IXA的,且只實現了MethodX,未實現MethodA

             

            struct CX : public IXA

            {

                virtual void MethodX()

                {

               

                }

            };

             

            然后CXA不再繼承IXA,而是直接繼承CX,補充實現MethodA

             

            struct CXA : public CX

            {

                virtual void MethodXA()

                {

               

                }

            };

             

            這樣,CXA是可以直接實例化的。但是,CX又無法單獨使用了!

             

            CX面臨的問題是,需要隨時被改變基類,也就是基類類型可配置。于是我們想到了模版:

             

            template <typename T = IX>

            struct CX : public T

            {

                virtual void MethodX()

                {

               

                }

            };

             

            struct CXA : public CX<IXA>

            {

                virtual void MethodXA()

                {

               

                }

            };

             

            CX單獨使用的時候,使用默認模版參數,讓它繼承于IX;被CXA使用的時候,讓它繼承于IXA,問題就解決了!同樣的方法,我們實現CXB

             

            struct CXB : public CX<IXB>

            {

                virtual void MethodXB()

                {

               

                }

            };

             

            下面我們來實現IXAB

             

            struct CXAB : public IXAB

            {

                virtual void MethodX()

                {

               

                }

             

                virtual void MethodXA()

                {

               

                }

             

                virtual void MethodXB()

                {

               

                }

             

                virtual void MethodXAB()

                {

               

                }

            };

             

            注意到這里只有一個MethodX。由于MethodXIXAIXB是同源的,IXAB里面雖然本應該有兩個MethodX,但編譯器進行合并處理,對其中一個做跳轉,所以我們只需要實現一次就夠了。(目前我是這樣理解的。)不過這不是本文要探討的內容,我們關注的是,MethodXMethodXAMethodXB又要重復寫一遍了。

             

            看來我們要對CXACXB也做模版處理:

             

            template <typename T = IXA>

            struct CXA : public CX<T>

            {

                virtual void MethodXA()

                {

               

                }

            };

             

            template <typename T = IXB>

            struct CXB : public CX<T>

            {

                virtual void MethodXB()

                {

               

                }

            };

             

            struct CXAB : public CXA<IXAB>, public CXB<IXAB>

            {

                virtual void MethodXAB()

                {

               

                }

            };

             

            可惜的是,還是有錯誤:

            error C2259: 'CXAB' : cannot instantiate abstract class

            due to following members:

            'void IXB::MethodXB(void)' : is abstract

            see declaration of 'IXB::MethodXB'

            'void IXA::MethodXA(void)' : is abstract

            see declaration of 'IXA::MethodXA'

             

            說的是CXA<IXAB>中的MethodXB未實現,CXB<IXAB>中的MethodXA未實現……所以,以上方案不適合接口中存在多繼承的情形。

             

            恰好COM中貌似也沒有接口多繼承的情形。我們將CX改名為IXImpl,將CXA改名為IXAImpl,將CXB改名為IXBImpl

             

            struct IX

            {

                virtual void MethodX() = 0;

            };

             

            struct IXA : public IX

            {

                virtual void MethodXA() = 0;

            };

             

            struct IXB : public IX

            {

                virtual void MethodXB() = 0;

            };

             

            template <typename T = IX>

            struct IXImpl : public T

            {

                virtual void MethodX()

                {

               

                }

            };

             

            template <typename T = IXA>

            struct IXAImpl : public IXImpl<T>

            {

                virtual void MethodXA()

                {

               

                }

            };

             

            template <typename T = IXB>

            struct IXBImpl : public IXImpl<T>

            {

                virtual void MethodXB()

                {

               

                }

            };

             

            是不是跟ATL對上號了?這些*Impl類可以直接使用(如果是需要默認實現就夠了),也可以被繼承,被繼承的時候也可以更換基類。如:

             

            struct CXA : public IXAImpl<>

            {

             

            };

             

            struct IY : public IXB

            {

                virtual void MethodY() = 0;

            };

             

            struct CY : public IXBImpl<IY>

            {

                virtual void MethodY()

                {

               

                }

            };

             

            第二個例子中,IXB是庫提供的接口,IY是應用程序設計的接口,CY是應用程序實現的類。

             

            好了,ATL中的*Impl類的做法就是如此,只是多了其他一些模版參數,暫時不理它們。

             

            posted on 2012-09-02 13:56 溪流 閱讀(2961) 評論(1)  編輯 收藏 引用 所屬分類: C++WindowsCOM

            評論:
            # re: 學習下 ATL 的繼承鏈處理 2012-09-04 00:30 | 畢達哥拉斯半圓
            good point!  回復  更多評論
              
            中文字幕人妻色偷偷久久| 亚洲精品无码专区久久久| 久久久国产精品网站| 久久人人爽人人精品视频| 日本久久中文字幕| 伊人久久综合无码成人网| 久久久青草青青亚洲国产免观| 色综合久久综精品| 久久国产免费直播| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 7国产欧美日韩综合天堂中文久久久久 | 久久久久久久综合综合狠狠| 日韩十八禁一区二区久久 | 亚洲精品久久久www| 99久久精品免费看国产一区二区三区 | 老色鬼久久亚洲AV综合| 久久精品无码一区二区日韩AV| 久久精品国产清自在天天线| 久久精品国产黑森林| 狠狠色婷婷综合天天久久丁香| 亚洲精品WWW久久久久久| 狠狠色伊人久久精品综合网 | 91精品国产91久久久久福利| 麻豆精品久久久久久久99蜜桃| 国产三级观看久久| 精品一区二区久久| 99精品久久精品| 精品久久久久久国产潘金莲| 亚洲精品美女久久久久99| 色婷婷噜噜久久国产精品12p| 久久国产成人午夜aⅴ影院 | 精品亚洲综合久久中文字幕| 国产精品美女久久久m| 久久人人爽人人爽人人片AV不 | 久久国产精品无| 亚洲人AV永久一区二区三区久久| 国内精品久久久久国产盗摄| 久久久久九九精品影院| 理论片午午伦夜理片久久 | 99久久亚洲综合精品成人| 久久久精品午夜免费不卡|