成員函式的過載、覆蓋(override)與隱藏很容易混淆,c++程式設計師必須要搞清楚概念,否則錯誤將防不勝防。
過載與覆蓋
成員函式被過載的特徵:
(1)相同的範圍(在同乙個類中);
(2)函式名字相同;
(3)引數不同;
(4)virtual 關鍵字可有可無。
覆蓋是指派生類函式覆蓋基類函式,特徵是:
(1)不同的範圍(分別位於派生類與基類);
(2)函式名字相同;
(3)引數相同;
(4)基類函式必須有virtual 關鍵字。
以下示例中,函式 base::f(int) 與 base::f(float) 相互過載,而 base::g(void) 被 derived::g(void) 覆蓋。
#include
using
namespace
std;
class base
void f(float x)
virtual
void g(void)
};class derived : public base
};int main(void)
令人迷惑的隱藏規則
本來僅僅區別過載與覆蓋並不算困難,但是c++的隱藏規則使問題複雜性陡然增加。
這裡「隱藏」是指派生類的函式遮蔽了與其同名的基類函式,規則如下:
(1)如果派生類的函式與基類的函式同名,但是引數不同。此時,不論有無virtual
關鍵字,基類的函式將被隱藏(注意別與過載混淆)。
(2)如果派生類的函式與基類的函式同名,並且引數也相同,但是基類函式沒有virtual
關鍵字。此時,基類的函式被隱藏(注意別與覆蓋混淆)。
以下示例中:
(1)函式derived::f(float) 覆蓋了 base::f(float)。
(2)函式derived::g(int) 隱藏了 base::g(float),而不是過載。
(3)函式derived::h(float) 隱藏了 base::h(float),而不是覆蓋。
#include
using
namespace
std;
class base
void g(float x)
void h(float x)
};class derived : public base
void g(int x)
void h(float x)
};
「隱藏」的發生可謂神出鬼沒,常常產生令人迷惑的結果。
如以下示例中,bp 和 dp 指向同一位址,按理說執行結果應該是相同的,可事實並非這樣。
int main(void)
擺脫隱藏
隱藏規則引起了不少麻煩。示例8-2-3 程式中,語句pd->f(10)的本意是想呼叫函式 base::f(int),但是 base::f(int) 不幸被 derived::f(char *str) 隱藏了。由於數字10不能被隱式地轉化為字串,所以在編譯時出錯。
class base
;class derived : public base
;void test(void)
c 成員函式的過載 覆蓋 隱藏區別
成員函式的過載 覆蓋 override 與隱藏很容易混淆,c 程式設計師必須要搞清楚概念,否則錯誤將防不勝防。過載與覆蓋 成員函式被過載的特徵 1 相同的範圍 在同乙個類中 2 函式名字相同 3 引數不同 4 virtual 關鍵字可有可無。覆蓋是指派生類函式覆蓋基類函式,特徵是 1 不同的範圍 分...
C 成員函式的過載 覆蓋 隱藏區別
一 過載與覆蓋 成員函式被過載的特徵 1 相同的範圍 在同乙個類中 2 函式名字相同 3 引數不同 4 virtual 關鍵字可有可無。覆蓋是指派生類函式覆蓋基類函式,特徵是 1 不同的範圍 分別位於派生類與基類 2 函式名字相同 3 引數相同 4 基類函式必須有virtual 關鍵字。二 隱藏規則...
C 成員函式的過載,覆蓋,隱藏
對於物件導向程式設計的三個特性 封裝,繼承,多型。封裝 封裝的目的主要是為了實現源 的安全性。所以我們對其有訪問控制。c 語言使用了三個明確的關鍵字來設定類中的邊界 public,private,protected。繼承的類可以訪問protected成員,但是不能訪問私有成員。繼承 重用介面,如果乙...