如下面的程序:
?1?????#include?<?iostream?>
?2?
?3?????using?namespace?std;
?4?
?5?????class?base{
?6?????public:
?7?????????virtual?void?f(char?i){
?8?????????????cout?<<?"Base?class:?"?<<?i?<<?endl;
?9?????????};
10?????????virtual?void?f(double?d){
11?????????????cout?<<?"Base?class:?"?<<?d?<<?endl;
12?????????}
13?????};
14?
15?????class?derived?:?public?base{
16?????public:
17?????????void?f(char?i){
18?????????????cout?<<?"Derived?class:?"?<<?i?<<?endl;
19?????????}
20?????};
21?
22?????int?main(int?argc,?char**?argv)
23?????{
24?????????base?*pb?=?new?derived;
25?????????pb->f(2.0);
26?????????delete?pb;
27?????????return?0;
28?????}
29?
開始的時候感覺有點迷惑,被調用的是 base 的 f 呢?還是 derived 的 f(需要進行參數類型轉換)。實驗證明,被調用的是 base 的 f。
因為重載是根據靜態類型來選擇函數的,亦即由 pb 本身的類型來決定。而虛擬則是根據動態類型來選擇函數,由 pb 指向的對象類型決定。因此編譯器首先根據重載來選擇一個基類的函數,如果該函數為虛擬函數且被派生類實現,則再通過多態機制調用派生類函數。