• <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>
            隨筆 - 74, 文章 - 0, 評論 - 26, 引用 - 0
            數(shù)據(jù)加載中……

            體驗C++中接口與實現(xiàn)分離的技術(shù)

            在用C++寫要導出類的庫時,我們經(jīng)常只想暴露接口,而隱藏類的實現(xiàn)細節(jié)。也就是說我們提供的頭文件里只提供要暴露的公共成員函數(shù)的聲明,類的其他所有信息都不會在這個頭文件里面顯示出來。這個時候就要用到接口與實現(xiàn)分離的技術(shù)。

              下面用一個最簡單的例子來說明。

              類ClxExp是我們要導出的類,其中有一個私有成員變量是ClxTest類的對象,各個文件內(nèi)容如下:

              lxTest.h文件內(nèi)容:

            class ClxTest
            {
             public:
              ClxTest();
              virtual ~ClxTest();
              void DoSomething();
            };

              lxTest.cpp文件內(nèi)容:

            #include "lxTest.h"

            #include <iostream>
            using namespace std;

            ClxTest::ClxTest()
            {}

            ClxTest::~ClxTest()
            {}

            void ClxTest::DoSomething()
            {
             cout << "Do something in class ClxTest!" << endl;
            }

            ////////////////////////////////////////////////////////////////////////////

              lxExp.h文件內(nèi)容:

            #include "lxTest.h"

            class ClxExp
            {
             public:
              ClxExp();
              virtual ~ClxExp();
              void DoSomething();
             private:
              ClxTest m_lxTest;
              void lxTest();
            };

              lxExp.cpp文件內(nèi)容:

            #include "lxExp.h"

            ClxExp::ClxExp()
            {}

            ClxExp::~ClxExp()
            {}

            // 其實該方法在這里并沒有必要,我這樣只是為了說明調(diào)用關(guān)系
            void ClxExp::lxTest()
            {
             m_lxTest.DoSomething();
            }

            void ClxExp::DoSomething()
            {
             lxTest();
            }

              為了讓用戶能使用我們的類ClxExp,我們必須提供lxExp.h文件,這樣類ClxExp的私有成員也暴露給用戶了。而且,僅僅提供lxExp.h文件是不夠的,因為lxExp.h文件include了lxTest.h文件,在這種情況下,我們還要提供lxTest.h文件。那樣ClxExp類的實現(xiàn)細節(jié)就全暴露給用戶了。另外,當我們對類ClxTest做了修改(如添加或刪除一些成員變量或方法)時,我們還要給用戶更新lxTest.h文件,而這個文件是跟接口無關(guān)的。如果類ClxExp里面有很多像m_lxTest那樣的對象的話,我們就要給用戶提供N個像lxTest.h那樣的頭文件,而且其中任何一個類有改動,我們都要給用戶更新頭文件。還有一點就是用戶在這種情況下必須進行重新編譯!

              上面是非常小的一個例子,重新編譯的時間可以忽略不計。但是,如果類ClxExp被用戶大量使用的話,那么在一個大項目中,重新編譯的時候我們就有時間可以去喝杯咖啡什么的了。當然上面的種種情況不是我們想看到的!你也可以想像一下用戶在自己程序不用改動的情況下要不停的更新頭文件和編譯時,他們心里會罵些什么。其實對用戶來說,他們只關(guān)心類ClxExp的接口DoSomething()方法。那我們怎么才能只暴露類ClxExp的DoSomething()方法而不又產(chǎn)生上面所說的那些問題呢?答案就是--接口與實現(xiàn)的分離。我可以讓類ClxExp定義接口,而把實現(xiàn)放在另外一個類里面。下面是具體的方法:

              首先,添加一個實現(xiàn)類ClxImplement來實現(xiàn)ClxExp的所有功能。注意:類ClxImplement有著跟類ClxExp一樣的公有成員函數(shù),因為他們的接口要完全一致。

              lxImplement.h文件內(nèi)容:

            #include "lxTest.h"

            class ClxImplement
            {
             public:
              ClxImplement();
              virtual ~ClxImplement();

              void DoSomething();
             
             private:
              ClxTest m_lxTest;
              void lxTest();
            };

              lxImplement.cpp文件內(nèi)容:

            #include "lxImplement.h"

            ClxImplement::ClxImplement()
            {}

            ClxImplement::~ClxImplement()
            {}

            void ClxImplement::lxTest()
            {
             m_lxTest.DoSomething();
            }

            void ClxImplement::DoSomething()
            {
             lxTest();
            }

              然后,修改類ClxExp。

              修改后的lxExp.h文件內(nèi)容:

            // 前置聲明
            class ClxImplement;

            class ClxExp
            {
             public:
              ClxExp();
              virtual ~ClxExp();
              void DoSomething();
             private:
              // 聲明一個類ClxImplement的指針,不需要知道類ClxImplement的定義
              ClxImplement *m_pImpl;
            };

              修改后的lxExp.cpp文件內(nèi)容:

            // 在這里包含類ClxImplement的定義頭文件
            #include "lxImplement.h"

            ClxExp::ClxExp()
            {
             m_pImpl = new ClxImplement;
            }

            ClxExp::~ClxExp()
            {
             delete m_pImpl;
            }

            void ClxExp::DoSomething()
            {
             m_pImpl->DoSomething();
            }

              通過上面的方法就實現(xiàn)了類ClxExp的接口與實現(xiàn)的分離。請注意兩個文件中的注釋。類ClxExp里面聲明的只是接口而已,而真正的實現(xiàn)細節(jié)被隱藏到了類ClxImplement里面。為了能在類ClxExp中使用類ClxImplement而不include頭文件lxImplement.h,就必須有前置聲明class ClxImplement,而且只能使用指向類ClxImplement對象的指針,否則就不能通過編譯。

              在發(fā)布庫文件的時候,我們只需給用戶提供一個頭文件lxExp.h就行了,不會暴露類ClxExp的任何實現(xiàn)細節(jié)。而且我們對類ClxTest的任何改動,都不需要再給用戶更新頭文件(當然,庫文件是要更新的,但是這種情況下用戶也不用重新編譯?。?。這樣做還有一個好處就是,可以在分析階段由系統(tǒng)分析員或者高級程序員來先把類的接口定義好,甚至可以把接口代碼寫好(例如上面修改后的lxExp.h文件和lxExp.cpp文件),而把類的具體實現(xiàn)交給其他程序員開發(fā)。

            posted on 2006-07-05 09:59 井泉 閱讀(193) 評論(0)  編輯 收藏 引用 所屬分類: C++

            狠狠干狠狠久久| 日本精品久久久久中文字幕| 国内精品久久久久久久coent| 91精品国产综合久久婷婷| 久久99国产综合精品免费| 亚洲国产成人久久精品影视| 91精品婷婷国产综合久久| 老司机午夜网站国内精品久久久久久久久 | 亚洲精品国产美女久久久| 久久棈精品久久久久久噜噜| 国产亚洲精久久久久久无码AV| 日韩欧美亚洲综合久久| 国产91久久精品一区二区| 国产精品99久久久精品无码| 狠狠色丁香久久综合婷婷| 久久无码AV中文出轨人妻| 亚洲欧美精品伊人久久| 中文字幕乱码人妻无码久久| 久久伊人精品青青草原高清| 中文字幕乱码人妻无码久久| 久久伊人色| 国产午夜精品久久久久九九| 97精品国产91久久久久久| 94久久国产乱子伦精品免费 | 亚洲国产成人精品91久久久 | 国产偷久久久精品专区| 久久久精品日本一区二区三区| 久久久噜噜噜久久熟女AA片| 久久丝袜精品中文字幕| 国产福利电影一区二区三区,免费久久久久久久精 | 国产精品青草久久久久福利99| 久久精品麻豆日日躁夜夜躁| 亚洲精品国产字幕久久不卡| 精品久久久久久国产| 伊人久久五月天| 伊人色综合久久天天人守人婷| 精品人妻伦一二三区久久| 久久精品国产亚洲一区二区| 69久久夜色精品国产69| 91精品国产91久久综合| 狠狠色丁香婷婷综合久久来|