讀書筆記 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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...