• <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>
            asm, c, c++ are my all
            -- Core In Computer
            posts - 139,  comments - 123,  trackbacks - 0

            /********************************************\
            |????歡迎轉(zhuǎn)載, 但請(qǐng)保留作者姓名和原文鏈接, 祝您進(jìn)步并共勉!???? |
            \********************************************/


            [譯著]在模板方法中的一些"反常"用法

            作者: Jerry Cat
            時(shí)間: 2006/05/19
            鏈接:?
            http://www.shnenglu.com/jerysun0818/archive/2006/05/19/7393.html

            -------------------------------------
            I. Virtually Yours -- Template Method模式
            我在研究Wendy寫的一個(gè)類。那是她為這個(gè)項(xiàng)目寫的一個(gè)抽象基類,而我的工作就是從中派生出一個(gè)具象類(concrete class)。這個(gè)類的public部分是這樣的:

            class Mountie {
            public:
            ??? void read( std::istream & );
            ??? void write( std::ostream & ) const;
            ??? virtual ~Mountie();

            很正常,virtual destructor表明這個(gè)類打算被繼承。那么再看看其protected部分:

            protected:
            ??? virtual void do_read( std::istream & );
            ??? virtual void do_write( std::ostream & ) const;

            也不過(guò)就是一會(huì)兒的功夫,我識(shí)破了Wendy的把戲:她在使用template method模式。public成員函數(shù)read和write是非虛擬的,它們肯定是調(diào)用protected部分do_read/do_write虛擬成員函數(shù)來(lái)完成實(shí)際的工作。啊,我簡(jiǎn)直為自己的進(jìn)步而飄飄然了!哈,Wendy,這回你可難不住我,還有什么招數(shù)?盡管放馬過(guò)來(lái)... 突然,笑容在我臉上凝固,因?yàn)槲铱吹搅似鋚rivate部分:

            private:
            ??? virtual std::string classID() const = 0;

            這是什么?一個(gè)private純虛函數(shù),能工作么?我站了起來(lái),

            “Wendy,你的Mountie類好像不能工作耶,它有一個(gè)private virtual function。”

            “你試過(guò)了?”她連頭都不抬。

            “嗯,那倒是沒(méi)有啦,可是想想也不行啊?我的派生類怎么能override你的private函數(shù)呢?” 我嘟囔著。

            “嗬,你倒是很確定啊!”Wendy的聲音很輕柔,“你怎么老是這也不行,那也不行的,這幾個(gè)月跟著我你就沒(méi)學(xué)到什么東西嗎?小菜鳥。”

            真是可惡啊...

            “小菜鳥,你全都忘了,訪問(wèn)控制級(jí)別跟一個(gè)函數(shù)是不是虛擬的根本沒(méi)關(guān)系。判斷一個(gè)函數(shù)是動(dòng)態(tài)綁定還是靜態(tài)綁定是函數(shù)調(diào)用解析的最后一個(gè)步驟。好好讀讀標(biāo)準(zhǔn)的3.4和5.2.2節(jié)吧。”

            我完全處于下風(fēng),只好采取干擾戰(zhàn)術(shù)。“好吧,就算你說(shuō)的不錯(cuò),我也還是不明白,何必把它設(shè)為private?”

            “我且問(wèn)你,倘若你不想讓一個(gè)類中的成員函數(shù)被其他的類調(diào)用,應(yīng)當(dāng)如何處理?”

            “當(dāng)然是把它設(shè)置為private的,” 我回答道。

            “那么你去看看我的Mountie類實(shí)現(xiàn),特別是write()函數(shù)的實(shí)現(xiàn)。”

            我正巴不得逃開(kāi)Wendy那刺人的目光,便轉(zhuǎn)過(guò)頭去在我的屏幕上搜索,很快,我找到了:

            void Mountie::write(std::ostream &Dudley) const
            {
            ??? Dudley << classID() << std::endl;
            ??? do_write(Dudley);
            }

            嗨,最近卡通片真是看得太多了,居然犯這樣的低級(jí)失誤。還是老是承認(rèn)吧:“好了,我明白了。classID()是一個(gè)實(shí)現(xiàn)細(xì)節(jié),用來(lái)在保存對(duì)象時(shí)指示具象類的類型,派生類必須覆蓋它,所以必須是純虛的。但是既然是實(shí)現(xiàn)細(xì)節(jié),就應(yīng)該設(shè)為private的。”

            “這還差不多,小菜鳥。”大蝦點(diǎn)了點(diǎn)頭,“現(xiàn)在給我解釋一下為什么do_read()和do_write()是protected的?”

            這個(gè)問(wèn)題并不難,我組織了一下就回答:“因?yàn)榕缮悓?duì)象需要調(diào)用這兩個(gè)函數(shù)的實(shí)現(xiàn)來(lái)讀寫其中的基類對(duì)象。”

            “很好很好,”大蝦差不多滿意了,“不過(guò),你再解釋解釋為什么我不把它們?cè)O(shè)為public的?”

            現(xiàn)在我感覺(jué)好多了:“因?yàn)檎{(diào)用它們的時(shí)候必須以一種特定的方式進(jìn)行。比如do_write()函數(shù),必須先把類型信息寫入,再把對(duì)象信息寫入,這樣讀取的時(shí)候,負(fù)責(zé)生成對(duì)象的模塊首先能夠知道要讀出來(lái)的對(duì)象是什么類型的,然后才能正確地從流中讀取對(duì)象信息。”

            “聰明啊,我的小菜鳥!”Wendy停頓了一下,“就跟學(xué)習(xí)外國(guó)口語(yǔ)一樣,學(xué)習(xí)C++也不光是掌握語(yǔ)法而已,還必須要掌握大量的慣用法。”

            “是啊是啊,我正打算讀Coplien的書...”

            [譯者注:就是James Coplien 1992年的經(jīng)典著作Advanced C++ Programming Style and Idioms]

            大蝦揮了揮她的手,“冷靜,小菜鳥,我不是指先知Coplien的那本書,我是指某種結(jié)構(gòu)背后隱含的慣用法。比如一個(gè)類有virtual destructor,相當(dāng)于告訴你說(shuō):‘嗨,我是一個(gè)多態(tài)基類,來(lái)繼承我吧!’ 而如果一個(gè)類的destructor不是虛擬的,則相當(dāng)于是在說(shuō):‘我不能作為多態(tài)基類,看在老天的份上,別繼承我。’”

            “同樣的,virtual函數(shù)的訪問(wèn)控制級(jí)別也具有隱含的意義。一個(gè)protected virtual function告訴你:‘你寫的派生類應(yīng)該,哦,可是說(shuō)是必須調(diào)用我的實(shí)現(xiàn)。’而一個(gè)private virtual function是在說(shuō):‘派生類可以覆蓋,也可以不覆蓋我,隨你的便。但是你不可以調(diào)用我的實(shí)現(xiàn)。’”

            我點(diǎn)點(diǎn)頭,告訴她我懂了,然后追問(wèn)道:“那么public virtual function呢?”

            “盡可能不要使用public virtual function。”她拿起一支筆寫下了以下代碼:

            class HardToExtend
            {
            public:
            ? virtual void f();
            };
            ?void HardToExtend::f()
            {
            ?// Perform a specific action
            }

            “假設(shè)你發(fā)布了這個(gè)類。在寫第二版時(shí),需求有所變化,你必須改用Template Method。可是這根本不可能,你知道為什么?”

            “呃,這個(gè)...,不知道。”

            “由兩種可能的辦法。其一,將f()的實(shí)現(xiàn)代碼轉(zhuǎn)移到一個(gè)新的函數(shù)中,然后將f()本身設(shè)為non-virtual的:

            class HardToExtend
            {
            // possibly protected
            ??? virtual void do_f();
            public:
            ??? void f();
            };
            void HardToExtend::f()
            {
            ??? // pre-processing
            ??? do_f();
            ??? // post-processing
            }
            void HardToExtend::do_f()
            {
            ??? // Perform a specific action
            }

            然而你原來(lái)寫的派生類都是企圖override函數(shù)f()而不是do_f()的,你必須改變所有的派生類實(shí)現(xiàn),只要你錯(cuò)過(guò)了一個(gè)類,你的類層次就會(huì)染上先知Meyers所說(shuō)的‘精神分裂的行徑’。” [譯者注:參見(jiàn)Scott Meyers,Effective C++, Item 37,絕對(duì)不要重新定義繼承而來(lái)的非虛擬函數(shù)]

            “另一種辦法是將f()移到private區(qū)域,引入一個(gè)新的non-virtual函數(shù):”

            class HardToExtend
            {
            // possibly protected
            ??? virtual void f();
            public:
            ??? void call_f();
            };

            “這會(huì)導(dǎo)致無(wú)數(shù)令人頭痛的問(wèn)題。首先,所有的客戶都企圖調(diào)用f()而不是call_f(),現(xiàn)在它們的代碼都不能編譯了。更有甚者,大部分派生類都回把f()放在public區(qū)域中,這樣直接使用派生類的用戶可以訪問(wèn)到你本來(lái)想保護(hù)的細(xì)節(jié)。”

            “對(duì)待虛函數(shù)要象對(duì)待數(shù)據(jù)成員一樣,把它們?cè)O(shè)為private的,直到設(shè)計(jì)上要求使用更寬松的訪問(wèn)控制再來(lái)調(diào)整。要知道由private入public易,由public入private難啊!”

            [譯者注:這篇文章所表達(dá)的思想具有一定的顛覆性,因?yàn)槲覀兲菀自诨愔性O(shè)置public virtual function了,Java中甚至專門為這種做法建立了interface機(jī)制,現(xiàn)在竟然說(shuō)這不好!一時(shí)間真是接受不了。但是仔細(xì)體會(huì)作者的意思,他并不是一般地反對(duì)public virtual function,只是在template method大背景下給出上述原則。雖然這個(gè)原則在一般的設(shè)計(jì)中也是值得考慮的,但是主要的應(yīng)用領(lǐng)域還是在template method模式中。當(dāng)然,template method是一種非常有用和常用的模式,因此也決定了本文提出的原則具有廣泛的意義。]

            posted on 2006-05-19 07:23 Jerry Cat 閱讀(669) 評(píng)論(1)  編輯 收藏 引用

            FeedBack:
            # re: [譯著]在模板方法中的一些"反常"用法
            2006-05-23 17:35 | cmdn
            請(qǐng)問(wèn)一下,這個(gè)系列的文章是從哪里傳過(guò)來(lái)的?給各地址 我指原版!  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理



            <2006年7月>
            2526272829301
            2345678
            9101112131415
            16171819202122
            23242526272829
            303112345

            常用鏈接

            留言簿(7)

            隨筆檔案

            最新隨筆

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            婷婷综合久久狠狠色99h| 久久强奷乱码老熟女| 久久久久女教师免费一区| 69国产成人综合久久精品| 日产精品99久久久久久| 久久无码专区国产精品发布| 美女久久久久久| 日批日出水久久亚洲精品tv| 久久影视综合亚洲| 开心久久婷婷综合中文字幕| 中文字幕精品久久| 国内精品久久久久影院薰衣草| 免费精品久久天干天干| 久久久久久久久久久精品尤物| 国产免费久久精品99re丫y| 亚洲国产成人精品无码久久久久久综合 | 久久综合狠狠综合久久激情 | 国产麻豆精品久久一二三| 久久九九亚洲精品| 欧美性大战久久久久久| 久久久久亚洲AV成人网人人网站 | 国产精品久久久久jk制服| 国产一久久香蕉国产线看观看| 亚洲国产天堂久久综合网站| 国产午夜精品久久久久九九| 无码任你躁久久久久久久| 亚洲精品美女久久久久99| 国产精品视频久久久| 久久精品成人欧美大片| 精品国产日韩久久亚洲| 国产∨亚洲V天堂无码久久久| 91精品国产色综久久| 18禁黄久久久AAA片| 97超级碰碰碰久久久久| 久久久久久av无码免费看大片| 久久综合亚洲鲁鲁五月天| 久久午夜电影网| 影音先锋女人AV鲁色资源网久久| 久久99热国产这有精品| 97精品伊人久久大香线蕉| 久久国产高潮流白浆免费观看|