c 過載 覆蓋 隱藏

2022-04-10 13:58:03 字數 3555 閱讀 5736

成員函式的過載、覆蓋與隱藏

成員函式的過載、覆蓋(override)與隱藏很容易混淆,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)中,bp 和dp 指向同一位址,按理說執行結果應該是相同的,可事

實並非這樣。

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 看來,隱藏規則似乎很愚蠢。但是隱藏規則至少有兩個存在的理由:

成員函式的過載、覆蓋與隱藏

成員函式的過載、覆蓋(override)與隱藏很容易混淆,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)中,bp 和dp 指向同一位址,按理說執行結果應該是相同的,可事

實並非這樣。

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 看來,隱藏規則似乎很愚蠢。但是隱藏規則至少有兩個存在的理由:

c 過載 覆蓋 隱藏

成員函式的過載 覆蓋與隱藏 成員函式的過載 覆蓋 override 與隱藏很容易混淆,c 程式設計師必須要搞清楚 概念,否則錯誤將防不勝防。8.2.1 過載與覆蓋 成員函式被過載的特徵 1 相同的範圍 在同乙個類中 2 函式名字相同 3 引數不同 4 virtual 關鍵字可有可無。覆蓋是指派生類函...

c 過載 覆蓋 隱藏

成員函式的過載 覆蓋與隱藏 成員函式的過載 覆蓋 override 與隱藏很容易混淆,c 程式設計師必須要搞清楚 概念,否則錯誤將防不勝防。8.2.1 過載與覆蓋 成員函式被過載的特徵 1 相同的範圍 在同乙個類中 2 函式名字相同 3 引數不同 4 virtual 關鍵字可有可無。覆蓋是指派生類函...

c 過載 覆蓋 隱藏

成員函式的過載 覆蓋與隱藏 成員函式的過載 覆蓋 override 與隱藏很容易混淆,c 程式設計師必須要搞清楚概念,否則錯誤將防不勝防。成員函式被過載的特徵 1 相同的範圍 在同乙個類中 2 函式名字相同 3 引數不同 4 virtual 關鍵字可有可無。覆蓋是指派生類函式覆蓋基類函式,特徵是 1...