人人做人人澡人人爽欧美,国产主播一区二区,久久久精品五月天,羞羞视频在线观看免费

當前位置:蘿卜系統下載站 > 技術開發教程 > 詳細頁面

C++ 與 Delphi 的函數覆蓋(Override)與重載(overload

C++ 與 Delphi 的函數覆蓋(Override)與重載(overload

更新時間:2022-05-30 文章作者:未知 信息來源:網絡 閱讀次數:

C++ 和 Delphi 的函數覆蓋(Override)與重載(overload)

Spacesoft【暗夜狂沙】

在面向對象編程中,當子類繼承了來自基類的函數后,子類有可能需要對其中的一些函數作出與基類不同處理,比如:

class CHuman
{
public:
void SayMyName()//打印出對象的姓名
{
cout << "Hi, I am a human" << endl;
}
};

那么很明顯,假如他的子類有一個同名、同參數和返回值(一句話,一摸一樣)的函數SayMyName,它會調用哪個函數呢?比如現在有一個class CMark

class CMark: public CHuman
{
public:
void SayMyName()
{
cout << "Hi, I am mark" << endl;
}
};

那么我們要問,下面的程序段:

CHuman *pH = new CMark;

if (pH)
pH->SayMyName();
else
cout << "cast error! " << endl;

delete pH;
pH = NULL;

要打印出來的,真的是我們想要的Hi, I am mark 嗎?

不是。它輸出了Hi, I am a human。這很糟糕,當我們指著一個人要他說出自己的名字的時候,他卻告訴我們他“是一個人”,而不是說出自己的名字。出現這樣的問題原因在于,用基類的指針指向公有派生類,可以訪問派生類從基類中繼承的成員函數。但如果派生類中也有同名的函數,則結果仍然是訪問基類的同名函數,而不是派生類本身的函數。而事實上,我們希望的是由一個對象的真實類型來決定到底該調用這些同名函數中的哪一個,就是說,這樣的決議是動態(Dynamic)的;蛘呶覀兛梢哉f,我們希望當一個對象是子類型時,它的同名函數在子類中的實現覆蓋(override)掉基類的實現。

我們先從C++對這個問題的處理說起。

這是C++中比較典型的多態的例子,C++用虛函數來實現這樣的多態。具體點說,就是使用virtual 關鍵字來將函數說明成虛函數,在上一個例子中就是應該聲明成:

class CHuman
{
public:
virtual void SayMyName()//打印出對象的姓名
{
cout << "Hi, I am a human" << endl;
}
};

這樣,其他的代碼還是那個老樣子,但是我們的CMark 已經知道怎么說自己的名字了。CMark 的SayMyName()函數是否加了virtual 關鍵字的說明并沒有關系,因為根據C++語法的規定,因為它覆蓋了CHuman 的同名函數,它自己也就成為virtual 的了。至于為什么一個virtual 關鍵字有那么神奇的效果呢?C++ FAQ Lite 對此是這樣說明的: 在C++中,“虛成員函數是動態確定的(在運行時)。也就是說,成員函數(在運行時)被動態地選擇,該選擇基于對象的類型,而不是指向該對象的指針/引用的類型”。于是我們的pH就發現自己其實指向的是一個CMark類型的對象,而不是自己的類型所聲明的CHuman,所以它聰明的調用了CMark的SayMyName。


而Delphi 就是用override 關鍵字來說明函數覆蓋的。被覆蓋的函數必須是虛(virtual)的,或者是動態(dynamic)的,也就是說該函數在聲明時應該包含這兩個指示字中的一個,比如:

procedure Draw; virtual;

在需要覆蓋的時候,只需要在子類中用override 指示字重新聲明一下就可以了。

procedure Draw; override;

在語法上來說,聲明為 virtual和 dynamic是等價的。它們的差別在于,前者在實現上對速度進行了優化,而后者對代碼大小進行了優化。

假如基類和子類都含有同一個函數名和參數,并且在子類中不加override 指示字呢?這在語法上也是正確的。這意味著子類的函數同名實現把基類的實現隱藏(hide)掉了,盡管這二者在派生類中都存在。那么就回到了本文開頭的第一個例子說明的情況:當我們指著一個人要他說出自己的名字的時候,他卻告訴我們他“是一個人”,而不是說出自己的名字。

值得注意的是,與我們在C++ 中常常不加區分的把覆蓋一個函數和重載一個函數通稱為重載不同,在Delphi 中,只有重載(overload) 才是我們平時所說的重載,被重載的函數依然存在,依靠參數來決定到底調用那個實現。當然,當overload掉的函數和基類的函數參數相同時,基類的實現就被hide掉了,就像上面提到的一樣。而覆蓋(override)則是把讓被覆蓋的函數不可見了,確確實實的"覆蓋"掉了,原來的實現就不見了;谶@樣的原因,許多文章甚至一些書都錯誤的把override翻譯成重載,筆者認為并不合適。

溫馨提示:喜歡本站的話,請收藏一下本站!

本類教程下載

系統下載排行

網站地圖xml | 網站地圖html
主站蜘蛛池模板: 项城市| 新晃| 南雄市| 道孚县| 连山| 海安县| 墨脱县| 宕昌县| 天门市| 镇赉县| 黄浦区| 进贤县| 福安市| 时尚| 银川市| 芒康县| 乐至县| 四川省| 伊川县| 沧州市| 贵州省| 永康市| 新巴尔虎左旗| 大厂| 大埔区| 广丰县| 灵丘县| 绥宁县| 察隅县| 海盐县| 祁东县| 沙湾县| 长春市| 苏州市| 华蓥市| 东方市| 绵竹市| 灵山县| 万源市| 广饶县| 平江县|