c++ 過載、覆蓋(override)與隱藏
稍微懂得點oop的人都知道過載,那是多型性的重要體現!可是在c++中你能分清成員函式的過載、覆蓋嗎?這個好像也不難,過載存在與同乙個類中,而覆蓋存在於派生類於基類中!可是如果再加上隱藏呢?說實話,以前我從來沒有聽說過這個概念!也不知道自己曾經捏造的程式,出了多少問題!看看林銳在《高質量c++程式設計指南》中的解釋。
8.2.1過載與覆蓋
成員函式被過載的特徵:
(1)相同的範圍(在同乙個類中);
(2)函式名字相同;
(3)引數不同;
(4)virtual關鍵字可有可無。
覆蓋是指派生類函式覆蓋基類函式,特徵是:
(1)不同的範圍(分別位於派生類與基類);
(2)函式名字相同;
(3)引數相同;
(4)基類函式必須有virtual關鍵字。
示例8-2-1中,函式base::f(int)與base::f(float)相互過載,而base::g(void)被derived::g(void)覆蓋。
#include
class base
void f(float x)
virtual void g(void) };
class derived : public base
};
void main(void)
示例8-2-1成員函式的過載和覆蓋
8.2.2令人迷惑的隱藏規則
本來僅僅區別過載與覆蓋並不算困難,但是c++的隱藏規則使問題複雜性陡然增加。這裡「隱藏」是指派生類的函式遮蔽了與其同名的基類函式,規則如下:
(1)如果派生類的函式與基類的函式同名,但是引數不同。此時,不論有無virtual關鍵字,基類的函式將被隱藏(注意別與過載混淆)。
(2)如果派生類的函式與基類的函式同名,並且引數也相同,但是基類函式沒有virtual關鍵字。此時,基類的函式被隱藏(注意別與覆蓋混淆)。
示例程式8-2-2(a)中:
(1)函式derived::f(float)覆蓋了base::f(float)。
(2)函式derived::g(int)隱藏了base::g(float),而不是過載。
(3)函式derived::h(float)隱藏了base::h(float),而不是覆蓋。
#include
class base
void g(float x)
void h(float x) };
class derived : public base
void g(int x)
void h(float x) };
示例8-2-2(a)成員函式的過載、覆蓋和隱藏
據作者考察,很多c++程式設計師沒有意識到有「隱藏」這回事。由於認識不夠深刻,「隱藏」的發生可謂神出鬼沒,常常產生令人迷惑的結果。
示例8-2-2(b)中,pb和pd指向同一位址,按理說執行結果應該是相同的,可事實並非這樣。
void main(void)
示例8-2-2(b) 過載、覆蓋和隱藏的比較
8.2.3擺脫隱藏
隱藏規則引起了不少麻煩。示例8-2-3程式中,語句pd->f(10)的本意是想呼叫函式base::f(int),但是base::f(int)不幸被derived::f(char *)隱藏了。由於數字10不能被隱式地轉化為字串,所以在編譯時出錯。
class base ;
class derived : public base ;
void test(void)
示例8-2-3 由於隱藏而導致錯誤
從示例8-2-3看來,隱藏規則似乎很愚蠢。但是隱藏規則至少有兩個存在的理由:
u 寫語句pd->f(10)的人可能真的想呼叫derived::f(char *)函式,只是他誤將引數寫錯了。有了隱藏規則,編譯器就可以明確指出錯誤,這未必不是好事。否則,編譯器會靜悄悄地將錯就錯,程式設計師將很難發現這個錯誤,流下禍根。
u 假如類derived有多個基類(多重繼承),有時搞不清楚哪些基類定義了函式f。如果沒有隱藏規則,那麼pd->f(10)可能會呼叫乙個出乎意料的基類函式f。儘管隱藏規則看起來不怎麼有道理,但它的確能消滅這些意外。
示例8-2-3中,如果語句pd->f(10)一定要呼叫函式base::f(int),那麼將類derived修改為如下即可。
class derived : public base
};
Overload與Override(覆蓋)的區別
過載方法 重寫方法 方法簽名引數列表必須修改 不能修改 返回型別可以修改 子類返回值 層次 父類返回值 層次 丟擲異常可以修改 子類丟擲異常範圍 父類丟擲異常範圍 方法許可權修飾符可以修改 子類許可權 父類許可權 測試 如下 parent類 public class parent child類 pu...
override與過載的區別
1 override與過載的區別 override修飾符,覆寫乙個基類中的方法.過載則是將將同名方法重新寫過,達到同名的函式實現不同的功能.從而實現了乙個方法有不同的版本.override與過載的本質區別是,加入了override的修飾符的方法,此方法始終只有乙個被你使用的方法.這個區別就好像 ov...
override與過載的區別
override與過載的區別 override修飾符,覆寫乙個基類中的方法.過載則是將將同名方法重新寫過,達到同名的函式實現不同的功能.從而實現了乙個方法有不同的版本.override與過載的本質區別是,加入了override的修飾符的方法,此方法始終只有乙個能被你使用的方法,而加入overload...