讀書筆記 effective c++ Item 37 永遠不要重新定義繼承而來的函數預設參數值

来源:http://www.cnblogs.com/harlanc/archive/2017/03/25/6617834.html
-Advertisement-
Play Games

從一開始就讓我們簡化這次的討論。你有兩類你能夠繼承的函數:虛函數和非虛函數。然而,重新定義一個非虛函數總是錯誤的(Item 36),所以我們可以安全的把這個條款的討論限定在繼承帶預設參數值的虛函數上。 1. 虛函數是動態綁定的,而預設參數是靜態綁定的 在這種情況下,這個條款的驗證就相當直接了:虛函數 ...


 

從一開始就讓我們簡化這次的討論。你有兩類你能夠繼承的函數:虛函數和非虛函數。然而,重新定義一個非虛函數總是錯誤的(Item 36),所以我們可以安全的把這個條款的討論限定在繼承帶預設參數值的虛函數上。

1. 虛函數是動態綁定的,而預設參數是靜態綁定的

在這種情況下,這個條款的驗證就相當直接了:虛函數是動態綁定的,而預設參數值是靜態綁定的。

這是什麼?你說你不堪重負的腦袋已經忘記了動態綁定和靜態綁定之間的區別?(為了好記,靜態綁定也叫做早綁定(early binding),動態綁定也叫做晚綁定(late binding))。讓我們看一下:

一個對象的靜態類型是你已經在程式文本中聲明的類型,考慮如下的類繼承體系:

 1 // a class for geometric shapes
 2 class Shape {
 3 public:
 4 enum ShapeColor { Red, Green, Blue };
 5 // all shapes must offer a function to draw themselves
 6 virtual void draw(ShapeColor color = Red) const = 0;
 7 ...
 8 };
 9 
10 class Rectangle: public Shape {
11 public:
12 // notice the different default parameter value — bad!
13 virtual void draw(ShapeColor color = Green) const;
14 ...
15 };
16 class Circle: public Shape {
17 public:
18 virtual void draw(ShapeColor color) const;
19 ...
20 };

 

畫成類繼承圖會是下麵這個樣子:

 

現在考慮三個指針:

1 Shape *ps;                               // static type = Shape*
2 
3 Shape *pc = new Circle;           // static type = Shape*
4 
5 Shape *pr = new Rectangle;    // static type = Shape*

 

在這個例子中,ps,pc和pr都被聲明為指向shape的指針,所以它們用Shape作為它們的靜態類型。註意無論shape指針真正指向的是什麼對象,靜態類型都是Shape*。

 

一個對象的動態類型由指針當前指向的對象類型來決定。也就是,它的動態類型表明瞭它的行為會是怎樣的。看上面的例子,pc的動態類型是Circle*,pr的動態類型是Rectangle*。對於ps,它實際上沒有動態類型,因為它還沒有引用任何對象。

 

正如字面意思所表示的,在程式運行時動態類型是可以改變的,特別是通過賦值:

1 ps = pc;      // ps’s dynamic type is now Circle* 
2 ps = pr;      // ps’s dynamic type is now Rectangle*

虛函數是動態綁定的,意味著哪個函數被調用是由發出調用的對象的動態類型來決定的:

1 pc->draw(Shape::Red);          // calls Circle::draw(Shape::Red)
2 
3 pr->draw(Shape::Red);          // calls Rectangle::draw(Shape::Red)

 

這些都是舊知識了,我知道你肯定瞭解虛函數。當你考慮帶預設參數值的虛函數時,麻煩出現了,因為虛函數是動態綁定的,但是預設參數是靜態綁定的。這意味著你可能會終止一個虛函數的調用,因為函數定義在派生類中卻使用了基類中的預設參數:

 

1  pr->draw();                           // calls Rectangle::draw(Shape::Red)!

 

在這種情況中,pr的動態類型是Rectangle*,所以Rectangle的虛函數被調用,這也是你所期望的。在Rectangle::draw中,預設參數值是Green。然而因為pr的靜態類型是Shape*,這個函數調用的預設參數值是來自於Shape類而不是Rectangle類!最後的結果是這個調用由一個奇怪的也幾乎是你意料不到的組合組成:也即是Shape類和Rectangle類中的draw聲明混合而成。

 

Ps,pc和pr都為指針不是造成這個問題的原因。如果它們是引用也同樣會出現這個問題。唯一重要的事情是draw是一個虛函數,並且預設參數中的一個在派生類中被重新定義了

 

2. C++為什麼不對參數進行動態綁定?

為什麼C++堅持用一種反常的方式來運行?答案和運行時效率相關。如果一個預設參數是動態綁定的,編譯器就需要用一種方法在運行時為虛函數參數確定一個合適的預設值,比起當前在編譯期決定這些參數的機制,它更慢更加複雜。做出的決定是更多的考慮了速度和實現的簡單性,結果是你可以享受高效的執行速度,但是如果你沒有註意到這個條款的建議,你就會很迷惑。

3. 個例討論——為基類和派生類提供相同的預設參數

 

這都很好,但是看看如果這麼做會發生什麼:遵守這個條款的規定並且為基類和派生類函數同時提供預設參數:

 1 class Shape {
 2 
 3 public:
 4 
 5 enum ShapeColor { Red, Green, Blue };
 6 
 7 virtual void draw(ShapeColor color = Red) const = 0;
 8 
 9 ...
10 
11 };
12 
13 class Rectangle: public Shape {
14 public:
15 virtual void draw(ShapeColor color = Red) const;
16 ...
17 };

 

代碼重覆的問題出現了。更糟糕的是,與代碼重覆問題便隨而來的代碼依賴問題:如果Shape中的預設參數被修改了,所有重覆這個參數的派生類都需要被修改。否則重新定義繼承而來的預設參數值的問題會再度出現。該怎麼做?

 

當你讓虛函數按照你的方式來運行時遇到了麻煩,考慮替代設計方法是很明智的,Item 35中介紹了替換虛函數的不同方法。其中的一個是非虛介面用法(NVI idiom):用基類中的public非虛函數調用一個private虛函數,private虛函數可以在派生類中重新被定義。現在,我們用非虛函數指定預設參數,而用虛函數來做實際的工作:

 1 class Shape {
 2 public:
 3 enum ShapeColor { Red, Green, Blue };
 4 
 5 void draw(ShapeColor color = Red) const   // now non-virtual
 6 
 7 {                                                                 
 8 
 9  
10 
11 doDraw(color);                            // calls a virtual
12 
13 }                                                  
14 
15 ...                                                 
16 
17 private:                                       
18 
19 
20 virtual void doDraw(ShapeColor color) const = 0; // the actual work is
21 }; // done in this func
22 class Rectangle: public Shape {
23 public:
24 ...
25 private:
26 
27 virtual void doDraw(ShapeColor color) const; // note lack of a
28 
29 ...                                                                       // default param val.
30 
31 };               

                                            

因為非虛函數應該永遠不會在派生類中被重定義(Item 36),這個設計保證draw的color預設參數應該永遠是Red。

 

4. 總結

永遠不要重新定義一個繼承而來的預設參數值,因為預設參數值是靜態綁定的,而虛函數——你應該重新定義的唯一的函數——是動態綁定的。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、準備所需的jar包 1.1所需jar包 1.Spring框架jar包 2.Mybatis框架jar包 3.Spring的AOP事務jar包 4.Mybatis整合Spring中間件jar包 5.aspectj框架jar包 6.aop聯盟jar包 7.資料庫驅動jar包 8.數據源c3p0所需ja ...
  • 額,,,, 前幾天,剛開始玩力熱實驗, 卻沒想到,平時愛玩的實驗誤差分析的不確定度竟然計算那麼複雜,連夜寫了一段代碼, (大佬勿噴,物理專業的小白剛自學,應該也沒人看。。。) 為了以後我用著方便,都寫成了函數塊,接下來會隨著實驗作業和Java的學習,繼續完善 (現在,不明白的是,兩組數,在定義的函數 ...
  • 此文章是基於 EasyUI+Knockout實現經典表單的查看、編輯 一. 準備工作 1. 點擊此下載相關文件,並把文件放到 ims 工程對應的文件夾下 二. 相關文件介紹 1. user.jsp:用戶管理界面 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 ...
  • Apache Thrift 是 Facebook 實現的一種高效的、支持多種編程語言的遠程服務調用的框架。 安裝thrift 從官網下載 然後按照./configure,make,make install 方式進行安裝,其他系統可以參照官網http://thrift.apache.org/docs/ ...
  • 下圖為測試結果: ...
  • 昨天沒有更新,特此來說明下原因,昨天回到家時已經甚晚,正逢公司這幾天項目比較緊張(bug多,趕需求,看著bug單齊刷刷的轉過來,心都顫抖了一下),沒有及時準備素材,今天又加了一天班(現在還在公司,偷個空隙趕緊發博,哈哈哈),所以昨晚沒有更博。 今天在改bug的時候發現瞭如圖的小問題,來分享一下,主要 ...
  • matplotlib實際上是一套面向對象的繪圖庫,它所繪製的圖表中的每個繪圖元素,例如線條Line2D、文字Text、刻度等在記憶體中都有一個對象與之對應。為了方便快速繪圖matplotlib通過pyplot模塊提供了一套和MATLAB類似的繪圖API,將眾多繪圖對象所構成的複雜結構隱藏在這套API內 ...
  • 轉自:http://blog.csdn.net/wushiwude/article/details/55101631 一、前言 dubbo的使用,其實只需要有註冊中心,消費者,提供者這三個就可以使用了,但是並不能看到有哪些消費者和提供者,為了更好的調試,發現問題,解決問題,因此引入dubbo-adm ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...