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

牽著老婆滿街逛

嚴以律己,寬以待人. 三思而后行.
GMail/GTalk: yanglinbo#google.com;
MSN/Email: tx7do#yahoo.com.cn;
QQ: 3 0 3 3 9 6 9 2 0 .

Effective C++ 2e Item39

來源:http://blog.csdn.net/lostmouse/archive/2001/07/31/8503.aspx

條款39: 避免 "向下轉換" 繼承層次

在當今喧囂的經濟時代,關注一下我們的金融機構是個不錯的主意。所以,看看下面這個有關銀行帳戶的協議類(Protocol class )(參見條款34):

 

class Person  };

class BankAccount {
public:
  BankAccount(
const Person *primaryOwner,
              
const Person *jointOwner);
  
virtual ~BankAccount();

  
virtual void makeDeposit(double amount) = 0;
  
virtual void makeWithdrawal(double amount) = 0;

  
virtual double balance() const = 0;

  

}
;

 

很多銀行現在提供了多種令人眼花繚亂的帳戶類型,但為簡化起見,我們假設只有一種銀行帳戶,稱為存款帳戶:

 

class SavingsAccount: public BankAccount {
public:
  SavingsAccount(
const Person *primaryOwner,
                 
const Person *jointOwner);
  
~SavingsAccount();

  
void creditInterest();                // 給帳戶增加利息

  

}
;

 

這遠遠稱不上是一個真正的存款帳戶,但還是那句話,現在什么年代?至少,它滿足我們現在的需要。

銀行想為它所有的帳戶維持一個列表,這可能是通過標準庫(參見條款49)中的list類模板實現的。假設列表被叫做allAccounts:

 

list<BankAccount*> allAccounts;         // 銀行中所有帳戶

 

和所有的標準容器一樣,list存儲的是對象的拷貝,所以,為避免每個BankAccount存儲多個拷貝,銀行決定讓allAccounts保存BankAccount的指針,而不是BankAccount本身。

假設現在準備寫一段代碼來遍歷所有的帳戶,為每個帳戶計算利息。你會這么寫:

 

// 不能通過編譯的循環(如果你以前從沒
// 見過使用 "迭代子" 的代碼,參見下文)
for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  (
*p)->creditInterest();      // 錯誤!

}

 

但是,編譯器很快就會讓你認識到:allAccounts包含的指針指向的是BankAccount對象,而非SavingsAccount對象,所以每次循環,p指向的是一個BankAccount。這使得對creditInterest的調用無效,因為creditInterest只是為SavingsAccount對象聲明的,而不是BankAccount。

如果"list<BankAccount*>::iterator p = allAccounts.begin()" 在你看來更象電話線中的噪音,而不是C++,那很顯然,你以前無緣見識過C++標準庫中的容器類模板。標準庫中的這一部分通常被稱為標準模板庫(STL),你可以在條款49和M35初窺其概貌。但現在你只用知道,變量p工作起來就象一個指針,它將allAccounts中的元素從頭到尾循環一遍。也就是說,p工作起來就好象它的類型是BankAccount**而列表中的元素都存儲在一個數組中。

上面的循環不能通過編譯很令人泄氣。的確,allAccounts是被定義為保存BankAccount*,但要知道,上面的循環中它事實上保存的是SavingsAccount*,因為SavingsAccount是僅有的可以被實例話的類。愚蠢的編譯器!對我們來說這么顯然的事情它竟然笨得一無所知。所以你決定告訴它:allAccounts真的包含的是SavingsAccount*:

 

// 可以通過編譯的循環,但很糟糕
for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  static_cast
<SavingsAccount*>(*p)->creditInterest();

}

 

一切問題迎刃而解!解決得很清晰,很漂亮,很簡明,所做的僅僅是一個簡單的轉換而已。你知道allAccounts指針保存的是什么類型的指針,遲鈍的編譯器不知道,所以你通過一個轉換來告訴它,還有比這更合理的事嗎?

在此,我要拿圣經的故事做比喻。轉換之于C++程序員,就象蘋果之于夏娃。

這種類型的轉換 ---- 從一個基類指針到一個派生類指針 ---- 被稱為 "向下轉換",因為它向下轉換了繼承的層次結構。在剛看到的例子中,向下轉換碰巧可以工作;但正如下面即將看到的,它將給今后的維護人員帶來惡夢。

還是回到銀行的話題上來。受到存款帳戶業務大獲成功的激勵,銀行決定再推出支票帳戶業務。另外,假設支票帳戶和存款帳戶一樣,也要負擔利息:

 

class CheckingAccount: public BankAccount {
public:
  
void creditInterest();    // 給帳戶增加利息

  

}
;

 

不用說,allAccounts現在是一個包含存款和支票兩種帳戶指針的列表。于是,上面所寫的計算利息的循環轉瞬間有了大麻煩。

第一個問題是,雖然新增了一個CheckingAccount,但即使不去修改循環代碼,編譯還是可以繼續通過。因為編譯器只是簡單地聽信于你所告訴它們(通過static_cast)的一切:*p指向的是SavingsAccount*。誰叫你是它的主人呢?這會給今后維護帶來第一個惡夢。維護期第二個惡夢在于,你一定想去解決這個問題,所以你會寫出這樣的代碼:

 

for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  
if (*p 指向一個 SavingsAccount)
    static_cast
<SavingsAccount*>(*p)->creditInterest();
  
else
    static_cast
<CheckingAccount*>(*p)->creditInterest();

}

 

任何時候發現自己寫出 "如果對象屬于類型T1,做某事;但如果屬于類型T2,做另外某事" 之類的代碼,就要扇自己一個耳光。這不是C++的做法。是的,在C,Pascal,甚至Smalltalk中,它是很合理的做法,但在C++中不是。在C++中,要使用虛函數。

記得嗎?對于一個虛函數,編譯器可以根據所使用對象的類型來保證正確的函數調用。所以不要在代碼中隨處亂扔條件語句或開關語句;讓編譯器來為你效勞。如下所示:

 

class BankAccount  };      // 同上

// 一個新類,表示要支付利息的帳戶
class InterestBearingAccount: public BankAccount {
public:
  
virtual void creditInterest() = 0;

  

}
;

class SavingsAccount: public InterestBearingAccount {

                             
// 同上

}
;

class CheckingAccount: public InterestBearingAccount {

                             
// as above

}
;

 

用圖形表示如下:

                         BankAccount
                                  ^
                                  |
                 InterestBearingAccount
                                 /\
                                /  \
                               /    \
     CheckingAccount   SavingsAccount

因為存款和支票賬戶都要支付利息,所以很自然地想到把這一共同行為轉移到一個公共的基類中。但是,如果假設不是所有的銀行帳戶都需要支付利息(以我的經驗,這當然是個合理的假設),就不能把它轉移到BankAccount類中。所以,要為BankAccount引入一個新的子類InterestBearingAccount,并使SavingsAccoun和CheckingAccount從它繼承。

存款和支票賬戶都要支付利息的事實是通過InterestBearingAccount的純虛函數creditInterest來體現的,它要在子類SavingsAccount和CheckingAccount中重新定義。

有了新的類層次結構,就可以這樣來重寫循環代碼:

 

// 好一些,但還不完美
for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  static_cast
<InterestBearingAccount*>(*p)->creditInterest();

}

 

盡管這個循環還是包含一個討厭的轉換,但代碼已經比過去健壯多了,因為即使又增加InterestBearingAccount新的子類到程序中,它還是可以繼續工作。

為了完全消除轉換,就必須對設計做一些改變。一種方法是限制帳戶列表的類型。如果能得到一列InterestBearingAccount對象而不是BankAccount對象,那就太好了:

 

// 銀行中所有要支付利息的帳戶
list<InterestBearingAccount*> allIBAccounts;

// 可以通過編譯且現在將來都可以工作的循環
for (list<InterestBearingAccount*>::iterator p =
        allIBAccounts.begin();
     p 
!= allIBAccounts.end();
     
++p) {

  (
*p)->creditInterest();

}

 

如果不想用上面這種 "采用更特定的列表" 的方法,那就讓creditInterest操作使用于所有的銀行帳戶,但對于不用支付利息的帳戶來說,它只是一個空操作。這個方法可以這樣來表示:

 

class BankAccount {
public:
  
virtual void creditInterest() {}

  

}
;

class SavingsAccount: public BankAccount  };
class CheckingAccount: public BankAccount  };
list
<BankAccount*> allAccounts;
// 看啊,沒有轉換!
for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  (
*p)->creditInterest();

}

 

要注意的是,虛函數BankAccount::creditInterest提供一個了空的缺省實現。這可以很方便地表示,它的行為在缺省情況下是一個空操作;但這也會給它本身帶來難以預見的問題。想知道內幕,以及如何消除這一危險,請參考條款36。還要注意的是,creditInterest是一個(隱式的)內聯函數,這本身沒什么問題;但因為它同時又是一個虛函數,內聯指令就有可能被忽略。條款33解釋了為什么。

正如上面已經看到的,"向下轉換" 可以通過幾種方法來消除。最好的方法是將這種轉換用虛函數調用來代替,同時,它可能對有些類不適用,所以要使這些類的每個虛函數成為一個空操作。第二個方法是加強類型約束,使得指針的聲明類型和你所知道的真的指針類型之間沒有出入。為了消除向下轉換,無論費多大工夫都是值得的,因為向下轉換難看、容易導致錯誤,而且使得代碼難于理解、升級和維護(參見條款M32)。

至此,我所說的都是事實;但,不是全部事實。有些情況下,真的不得不執行向下轉換。

例如,假設還是面臨本條款開始的那種情況,即,allAccounts保存BankAccount指針,creditInterest只是為SavingsAccount對象定義,要寫一個循環來為每個帳戶計算利息。進一步假設,你不能改動這些類;你不能改變BankAccount,SavingsAccount或allAccounts的定義。(如果它們在某個只讀的庫中定義,就會出現這種情況)如果是這樣的話,你就只有使用向下轉換了,無論你認為這個辦法有多丑陋。

盡管如此,還是有比上面那種原始轉換更好的辦法。這種方法稱為 "安全的向下轉換",它通過C++的dynamic_cast運算符(參見條款M2)來實現。當對一個指針使用dynamic_cast時,先嘗試轉換,如果成功(即,指針的動態類型(見條款38)和正被轉換的類型一致),就返回新類型的合法指針;如果dynamic_cast失敗,返回空指針。

下面就是加上了 "安全向下轉換" 的例子:

 

class BankAccount  };          // 和本條款開始時一樣

class SavingsAccount:               // 同上
  public BankAccount  };

class CheckingAccount:              // 同上
  public BankAccount  };

list
<BankAccount*> allAccounts;     // 看起來應該熟悉些了吧

void error(const string& msg);      // 出錯處理函數;
                                    
// 見下文

// 嗯,至少轉換很安全
for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  
// 嘗試將*p安全轉換為SavingsAccount*;
  
// psa的定義信息見下文
  if (SavingsAccount *psa =
        dynamic_cast
<SavingsAccount*>(*p)) {
    psa
->creditInterest();
  }


  
// 嘗試將它安全轉換為CheckingAccount
  else if (CheckingAccount *pca =
             dynamic_cast
<CheckingAccount*>(*p)) {
    pca
->creditInterest();
  }


  
// 未知的帳戶類型
  else {
    error(
"Unknown account type!");
  }

}

 

這種方法遠不夠理想,但至少可以檢測到轉換失敗,而用dynamic_cast是無法做到的。但要注意,對所有轉換都失敗的情況也要檢查。這正是上面代碼中最后一個else語句的用意所在。采用虛函數,就不必進行這樣的檢查,因為每個虛函數調用必然都會被解析為某個函數。然而,一旦打算進行轉換,這一切好處都化為烏有。例如,如果某個人在類層次結構中增加了一種新類型的帳戶,但又忘了更新上面的代碼,所有對它的轉換就會失敗。所以,處理這種可能發生的情況十分重要。大部分情況下,并非所有的轉換都會失敗;但是,一旦允許轉換,再好的程序員也會碰上麻煩。

上面if語句的條件部分,有些看上去象變量定義的東西,看到它你是不是慌張地擦了擦眼鏡?如果真這樣,別擔心,你沒看錯。這種定義變量的方法是和dynamic_cast同時增加到C++語言中的。這一特性使得寫出的代碼更簡潔,因為對psa或pca來說,它們只有在被dynamic_cast成功初始化的情況下,才會真正被用到;使用新的語法,就不必在(包含轉換的)條件語句外定義這些變量。(條款32解釋了為什么通常要避免多余的變量定義)如果編譯器尚不支持這種定義變量的新方法,可以按老方法來做:

 

for (list<BankAccount*>::iterator p = allAccounts.begin();
     p 
!= allAccounts.end();
     
++p) {

  SavingsAccount 
*psa;        // 傳統定義
  CheckingAccount *pca;       // 傳統定義

  
if (psa = dynamic_cast<SavingsAccount*>(*p)) {
    psa
->creditInterest();
  }


  
else if (pca = dynamic_cast<CheckingAccount*>(*p)) {
    pca
->creditInterest();
  }


  
else {
    error(
"Unknown account type!");
  }

}

 

當然,從處理事情的重要性來說,把psa和pca這樣的變量放在哪兒定義并不十分重要。重要之處在于:用if-then-else風格的編程來進行向下轉換比用虛函數要遜色得多,應該將這種方法保留到萬不得已的情況下使用。運氣好的話,你的程序世界里將永遠看不到這樣悲慘荒涼的景象。



Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=8503

posted on 2007-08-30 01:44 楊粼波 閱讀(227) 評論(2)  編輯 收藏 引用

評論

# re: Effective C++ 2e Item39 2007-08-30 18:14 junyan

不容易涅。。。。一般的做帳都不容易了,還要編成程序。。??植腊?。。。加油~~飄過~!*_*  回復  更多評論   

# re: Effective C++ 2e Item39 [未登錄] 2007-08-30 23:55 楊粼波

恩恩。。。  回復  更多評論   


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


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美日韩在线视频观看| 国产精品大片免费观看| 一区二区三区视频在线观看| 日韩午夜免费| 午夜亚洲一区| 欧美国产日韩一区| 亚洲黄色在线| 欧美mv日韩mv亚洲| 欧美激情亚洲视频| 亚洲日本久久| 亚洲一区二区三区中文字幕| 亚洲青色在线| 欧美私人网站| 亚洲在线视频观看| 亚洲人体1000| 夜色激情一区二区| 99亚洲一区二区| 亚洲精品日韩在线| 亚洲视频成人| 久久久久久成人| 亚洲国产一区二区三区a毛片| 欧美激情视频给我| av成人免费在线观看| 亚洲欧美影院| 欧美成年网站| 亚洲三级性片| 国产精品电影在线观看| 亚洲特黄一级片| 亚洲欧美日韩区| 免费在线国产精品| 欧美视频在线观看 亚洲欧| 国产酒店精品激情| 欧美激情亚洲精品| 久久露脸国产精品| 久久精品国产亚洲aⅴ| 亚洲国产欧美日韩| 中国日韩欧美久久久久久久久| 欧美一级日韩一级| 欧美精品v日韩精品v国产精品| 一区二区免费在线观看| 蜜臀久久99精品久久久久久9| aa成人免费视频| 性欧美videos另类喷潮| 亚洲国产美女| 亚洲一区二区三区乱码aⅴ| 一区免费观看视频| 9久re热视频在线精品| 国产午夜久久久久| 亚洲精品专区| 一区二区视频免费在线观看| 99国产精品99久久久久久粉嫩| 亚洲二区在线观看| 久久久999精品视频| 91久久久久| 国产精品视频自拍| 亚洲国产精品成人| 国内一区二区三区| 99伊人成综合| 99精品国产福利在线观看免费| 欧美亚洲综合在线| 亚洲午夜精品在线| 欧美福利视频在线| 欧美va亚洲va香蕉在线| 国产精品国产三级国产aⅴ9色| 亚洲大胆人体在线| 美女91精品| 国产精品一区二区久久国产| 最新日韩av| 国产区亚洲区欧美区| 亚洲精品视频免费| 日韩视频在线观看一区二区| 牛夜精品久久久久久久99黑人| 另类尿喷潮videofree| 国内精品久久久久国产盗摄免费观看完整版| 一区二区日韩欧美| 午夜久久福利| 国产综合色产在线精品| 亚洲国产精品久久人人爱蜜臀 | 亚洲性感美女99在线| 欧美美女福利视频| 亚洲一区二区三区高清| 欧美一区二区视频观看视频| 国产婷婷97碰碰久久人人蜜臀| 久久国产欧美| 亚洲欧洲精品一区二区三区 | 午夜精品久久一牛影视| 国产区精品在线观看| 久久亚洲综合网| 亚洲激情一区二区| 欧美中文字幕在线视频| 亚洲第一区中文99精品| 欧美日韩精品一区二区三区四区| 在线视频精品一| 免费观看欧美在线视频的网站| 亚洲理伦电影| 国产视频一区二区三区在线观看| 久久久精品国产免大香伊| 欧美激情一区在线| 欧美亚洲尤物久久| 日韩视频在线观看| 亚洲国产精品www| 国产日韩在线看片| 欧美日韩天堂| 久热成人在线视频| 性欧美在线看片a免费观看| 亚洲欧美精品suv| 亚洲精品一级| 在线观看日产精品| 国产真实精品久久二三区| 欧美日韩一区二区三区四区在线观看| 午夜精品一区二区三区在线视| 最近中文字幕mv在线一区二区三区四区| 午夜久久久久久久久久一区二区| 亚洲国内在线| 亚洲二区三区四区| 国产麻豆午夜三级精品| 欧美日韩在线视频一区| 欧美日韩mv| 欧美日韩在线一区| 欧美三日本三级少妇三2023| 久久亚洲视频| 久久婷婷国产综合尤物精品| 久久精品国产免费| 亚洲私人黄色宅男| 国模吧视频一区| 国产日产亚洲精品系列| 国产美女精品人人做人人爽| 欧美日韩一区二区视频在线观看| 男女精品网站| 欧美精品日韩综合在线| 欧美日韩中国免费专区在线看| 欧美日韩国产999| 国产精品99免视看9| 欧美先锋影音| 国产欧美日本一区二区三区| 国产一区二区三区四区| 在线看欧美视频| 9l国产精品久久久久麻豆| 亚洲一区二区三区乱码aⅴ| 香蕉久久一区二区不卡无毒影院| 久久精品视频一| 亚洲国产精品久久| 亚洲一区二区三区国产| 久久久久久久久久久久久女国产乱 | 一色屋精品视频在线看| 亚洲激情黄色| 久久gogo国模啪啪人体图| 亚洲电影激情视频网站| 亚洲图片欧美一区| 免费人成网站在线观看欧美高清| 欧美性猛交xxxx乱大交蜜桃| 国产一区二区三区黄| 亚洲线精品一区二区三区八戒| 蜜桃av噜噜一区| 先锋影音久久| 国产精品国产三级国产普通话99 | 久久精品视频va| 最新中文字幕亚洲| 久久亚洲图片| 狠狠狠色丁香婷婷综合激情| 亚洲五月六月| 亚洲激情av| 蜜桃精品久久久久久久免费影院| 欧美一区免费| 国产欧美午夜| 亚洲欧美精品| 在线视频亚洲| 欧美亚洲成人精品| 一本色道久久加勒比88综合| 欧美xxxx在线观看| 久久一区二区三区四区| 国产日韩欧美二区| 欧美一区二区三区婷婷月色| 最新国产乱人伦偷精品免费网站 | 亚洲精品在线观| 欧美日韩国产电影| 99视频热这里只有精品免费| 亚洲欧洲精品一区二区| 欧美a级片一区| 一本到12不卡视频在线dvd| 亚洲国产中文字幕在线观看| 欧美激情一区二区三区在线视频观看 | 一本久久知道综合久久| 欧美日韩岛国| 亚洲在线免费观看| 欧美在线视频观看免费网站| 好看的日韩av电影| 91久久国产综合久久蜜月精品| 欧美色区777第一页| 久久精品日产第一区二区| 蜜臀久久久99精品久久久久久 | 亚洲欧美激情视频| 在线观看亚洲视频| 亚洲毛片在线| 精品96久久久久久中文字幕无| 亚洲欧洲日产国产网站| 国产一区二区三区的电影 | 国产精品女主播| 欧美成年人视频| 国产精品你懂的在线欣赏|