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

asm, c, c++ are my all
-- Core In Computer
posts - 139,  comments - 123,  trackbacks - 0

/********************************************\
|????歡迎轉載, 但請保留作者姓名和原文鏈接, 祝您進步并共勉!???? |
\********************************************/


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

作者: Jerry Cat
時間: 2006/05/19
鏈接:?
http://m.shnenglu.com/jerysun0818/archive/2006/05/19/7393.html

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

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

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

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

也不過就是一會兒的功夫,我識破了Wendy的把戲:她在使用template method模式。public成員函數read和write是非虛擬的,它們肯定是調用protected部分do_read/do_write虛擬成員函數來完成實際的工作。啊,我簡直為自己的進步而飄飄然了!哈,Wendy,這回你可難不住我,還有什么招數?盡管放馬過來... 突然,笑容在我臉上凝固,因為我看到了其private部分:

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

這是什么?一個private純虛函數,能工作么?我站了起來,

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

“你試過了?”她連頭都不抬。

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

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

真是可惡啊...

“小菜鳥,你全都忘了,訪問控制級別跟一個函數是不是虛擬的根本沒關系。判斷一個函數是動態綁定還是靜態綁定是函數調用解析的最后一個步驟。好好讀讀標準的3.4和5.2.2節吧。”

我完全處于下風,只好采取干擾戰術。“好吧,就算你說的不錯,我也還是不明白,何必把它設為private?”

“我且問你,倘若你不想讓一個類中的成員函數被其他的類調用,應當如何處理?”

“當然是把它設置為private的,” 我回答道。

“那么你去看看我的Mountie類實現,特別是write()函數的實現。”

我正巴不得逃開Wendy那刺人的目光,便轉過頭去在我的屏幕上搜索,很快,我找到了:

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

嗨,最近卡通片真是看得太多了,居然犯這樣的低級失誤。還是老是承認吧:“好了,我明白了。classID()是一個實現細節,用來在保存對象時指示具象類的類型,派生類必須覆蓋它,所以必須是純虛的。但是既然是實現細節,就應該設為private的。”

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

這個問題并不難,我組織了一下就回答:“因為派生類對象需要調用這兩個函數的實現來讀寫其中的基類對象。”

“很好很好,”大蝦差不多滿意了,“不過,你再解釋解釋為什么我不把它們設為public的?”

現在我感覺好多了:“因為調用它們的時候必須以一種特定的方式進行。比如do_write()函數,必須先把類型信息寫入,再把對象信息寫入,這樣讀取的時候,負責生成對象的模塊首先能夠知道要讀出來的對象是什么類型的,然后才能正確地從流中讀取對象信息。”

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

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

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

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

“同樣的,virtual函數的訪問控制級別也具有隱含的意義。一個protected virtual function告訴你:‘你寫的派生類應該,哦,可是說是必須調用我的實現。’而一個private virtual function是在說:‘派生類可以覆蓋,也可以不覆蓋我,隨你的便。但是你不可以調用我的實現。’”

我點點頭,告訴她我懂了,然后追問道:“那么public virtual function呢?”

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

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

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

“呃,這個...,不知道。”

“由兩種可能的辦法。其一,將f()的實現代碼轉移到一個新的函數中,然后將f()本身設為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
}

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

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

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

“這會導致無數令人頭痛的問題。首先,所有的客戶都企圖調用f()而不是call_f(),現在它們的代碼都不能編譯了。更有甚者,大部分派生類都回把f()放在public區域中,這樣直接使用派生類的用戶可以訪問到你本來想保護的細節。”

“對待虛函數要象對待數據成員一樣,把它們設為private的,直到設計上要求使用更寬松的訪問控制再來調整。要知道由private入public易,由public入private難啊!”

[譯者注:這篇文章所表達的思想具有一定的顛覆性,因為我們太容易在基類中設置public virtual function了,Java中甚至專門為這種做法建立了interface機制,現在竟然說這不好!一時間真是接受不了。但是仔細體會作者的意思,他并不是一般地反對public virtual function,只是在template method大背景下給出上述原則。雖然這個原則在一般的設計中也是值得考慮的,但是主要的應用領域還是在template method模式中。當然,template method是一種非常有用和常用的模式,因此也決定了本文提出的原則具有廣泛的意義。]

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

FeedBack:
# re: [譯著]在模板方法中的一些"反常"用法
2006-05-23 17:35 | cmdn
請問一下,這個系列的文章是從哪里傳過來的?給各地址 我指原版!  回復  更多評論
  

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理



<2006年8月>
303112345
6789101112
13141516171819
20212223242526
272829303112
3456789

常用鏈接

留言簿(7)

隨筆檔案

最新隨筆

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲欧美日韩久久精品| 国产日韩精品入口| 亚洲欧美激情视频在线观看一区二区三区 | 在线午夜精品自拍| 亚洲午夜国产一区99re久久| 亚洲一区一卡| 欧美在线视频一区| 欧美电影在线观看| 国产精品高潮呻吟久久av无限 | 一区在线电影| 亚洲精品国产精品国自产观看浪潮 | 欧美久久久久久蜜桃| 欧美精品亚洲精品| 国产精品久久久久久久久免费| 国产精品乱码一区二三区小蝌蚪| 国产一区二区三区在线免费观看| 亚洲韩国日本中文字幕| 亚洲电影下载| 午夜在线不卡| 一区二区在线视频播放| 亚洲国产精品成人| 亚洲欧美精品中文字幕在线| 欧美自拍偷拍午夜视频| 亚洲高清在线视频| 99精品视频免费观看视频| 欧美一区高清| 欧美日韩高清在线观看| 国产美女精品| 99re6这里只有精品| 久久精品夜夜夜夜久久| 亚洲精品日韩久久| 久久久久五月天| 国产精品国产三级国产aⅴ9色| 精品电影一区| 午夜亚洲伦理| 亚洲精品视频在线观看网站| 欧美亚洲一区| 欧美四级在线| 亚洲精品乱码久久久久久蜜桃91| 亚洲欧美综合另类中字| 亚洲精品日产精品乱码不卡| 久久夜色精品国产噜噜av| 国产精品欧美日韩一区二区| 亚洲最新合集| 欧美好吊妞视频| 久久高清国产| 国产日韩精品一区二区三区在线 | 久久天天躁狠狠躁夜夜爽蜜月| 亚洲国产一区二区精品专区| 欧美中文在线观看| 国产人妖伪娘一区91| 在线一区二区日韩| 亚洲国产99精品国自产| 久久精品国产亚洲高清剧情介绍| 国产精品国产三级国产aⅴ无密码| 亚洲毛片在线看| 亚洲国产第一页| 牛牛国产精品| 亚洲三级视频在线观看| 亚洲国产一区二区精品专区| 欧美激情片在线观看| 亚洲精品美女91| 欧美激情四色| 欧美裸体一区二区三区| 日韩亚洲欧美中文三级| 亚洲日本中文| 国产精品扒开腿做爽爽爽视频| 亚洲最新在线| 亚洲免费影视| 黄色亚洲免费| 一区二区欧美日韩| 亚洲精品一区二区网址 | 久久久久久999| 一区二区欧美日韩| 国产精品福利网| 欧美亚洲免费高清在线观看| 亚洲日本欧美天堂| 欧美精品www| 一区二区三区蜜桃网| 在线亚洲精品福利网址导航| 国产精品一区久久久久| 久久精品国产一区二区电影| 久久久99精品免费观看不卡| 91久久国产综合久久蜜月精品| 亚洲精品国产视频| 国产精品午夜国产小视频| 久久精品一区二区国产| 模特精品裸拍一区| 亚洲一区二区三区精品在线| 欧美在线播放高清精品| 亚洲精品乱码久久久久久黑人 | 亚洲看片一区| 亚洲人成人一区二区三区| 欧美午夜不卡影院在线观看完整版免费| 亚洲欧美偷拍卡通变态| 久久久久一区二区三区| 亚洲视频欧美视频| 久久精品视频导航| 亚洲少妇中出一区| 久久一区二区精品| 中国日韩欧美久久久久久久久| 亚洲亚洲精品三区日韩精品在线视频 | 亚洲图片欧洲图片av| 精品盗摄一区二区三区| 一本一道久久综合狠狠老精东影业| 国产在线精品成人一区二区三区| 美女91精品| 国产精品伦理| 亚洲风情在线资源站| 国产欧美亚洲一区| 一区二区精品国产| 亚洲精品欧洲精品| 久久婷婷国产综合精品青草| 在线视频欧美一区| 欧美aa在线视频| 老司机午夜精品视频在线观看| 欧美高清视频一区二区| 久久综合综合久久综合| 国产精品视频xxxx| 亚洲国产成人精品女人久久久| 国产综合精品| 亚洲精品视频免费| 久久久久久久尹人综合网亚洲| 欧美日本成人| 欧美不卡福利| 精品福利电影| 欧美在线二区| 久久成人精品一区二区三区| 欧美精品久久久久久| 亚洲大胆女人| 亚洲黄一区二区三区| 久久综合九色九九| 米奇777超碰欧美日韩亚洲| 韩日精品视频一区| 久久精品一区二区| 欧美成人午夜视频| 亚洲黄色精品| 欧美国产激情二区三区| 亚洲国产日韩欧美| 日韩一二三在线视频播| 欧美精品色综合| 亚洲精品一区二区三区福利| 在线视频日本亚洲性| 国产精品久久久久久超碰| 亚洲中字在线| 久久久亚洲精品一区二区三区 | 亚洲国产高清一区| 99精品热视频| 国产精品hd| 香蕉成人啪国产精品视频综合网| 欧美在线观看日本一区| 国产一区二区三区精品久久久| 欧美一区二区日韩| 欧美韩国日本一区| 亚洲在线观看视频| 国产日韩视频| 每日更新成人在线视频| 91久久久久久久久| 午夜伦理片一区| 又紧又大又爽精品一区二区| 免费国产自线拍一欧美视频| 亚洲黄色免费电影| 欧美一二三区在线观看| 极品少妇一区二区三区精品视频| 久久久综合视频| 99国内精品久久| 久久久xxx| 99国产精品99久久久久久| 国产精品久久久久久久久久久久久 | 久久精品国产综合| 亚洲精品国产品国语在线app| 亚洲欧美日韩综合国产aⅴ| 狠狠色噜噜狠狠色综合久| 欧美激情一级片一区二区| 午夜精品久久久久久久蜜桃app| 麻豆久久久9性大片| 亚洲午夜视频| 在线免费观看一区二区三区| 欧美精品电影在线| 久久久综合免费视频| 亚洲一区亚洲二区| 亚洲综合色在线| 久久久久久久久久码影片| 亚洲精品免费在线| 久久精品一二三区| 亚洲欧美日韩国产中文在线| 亚洲国产成人高清精品| 国产情人节一区| 欧美日韩三级| 免费不卡亚洲欧美| 欧美在线日韩精品| 亚洲一级黄色av| 日韩网站在线观看| 亚洲电影在线免费观看| 久久综合狠狠综合久久综合88| 亚洲资源av| 亚洲一区二区三区在线看| 亚洲人成绝费网站色www| 国产亚洲观看| 国产欧美激情|