轉(zhuǎn)載來自:http://blog.csdn.net/cadcisdhht/article/category/785138
總結(jié) :
?static 的作用是表示該函數(shù)只作用在類型的靜態(tài)變量上,與類的實例沒有關系;而 const 的作用是確保函數(shù)不能修改類的實例的狀態(tài),與類型的靜態(tài)變量沒有關系。因此不能同時用它們。 ?空類實例不包含信息,本來求 sizeof 應該是 0 。但當我們聲明該類型的實例的時候,它必須在內(nèi)存中占有一定的空間,否則無法使用這些實例。至于占用多少內(nèi)存,由編譯器決定。 Visual Studio 2008 中每個空類型的實例占用一個 byte 的空間。 由于構(gòu)造函數(shù)和析構(gòu)函數(shù)的調(diào)用與類型的實例無關(調(diào)用它們只需要知道函數(shù)地址即可),在它的實例中不需要增加任何信息。C++ 的編譯器一旦發(fā)現(xiàn)一個類型中有虛擬函數(shù),就會為該類型生成虛函數(shù)表,并在該類型的每一個實例中添加一個指向虛函數(shù)表的指針。在 32 位的機器上,一個指針占 4 個字節(jié)的空間。 ?每個非靜態(tài)類成員函數(shù)都隱藏一個指向自己實例的this指針。 ? 調(diào)用虛函數(shù)需要從一個實例中指向虛函數(shù)表的指針以得到函數(shù)的地址,因此調(diào)用虛函數(shù)需要一個實例。調(diào)用靜態(tài)成員函數(shù)不要實例。 ? &(pPoint->z) 的語意是求 pPoint 中變量 z 的地址( pPoint 的地址 0 加 z 的偏移量 8 ),并不需要訪問 pPoint 指向的內(nèi)存。 ? 如果沒有標明函數(shù)或者變量是的訪問權(quán)限級別,在 struct 中,是 public 的;而在 class 中,是 private 的。 ? 在 C++ 中,成員變量的初始化順序與變量在類型中的申明順序相同,而與它們在構(gòu)造函數(shù)的初始化列表中的順序無關。 ? 如果允許復制構(gòu)造函數(shù)傳值,那么會形成永無休止的遞歸并造成棧溢出,編譯錯誤 。因此 C++ 的標準不允許復制構(gòu)造函數(shù)傳值參數(shù),而必須是傳引用或者常量引用。 ? sizeof(數(shù)組) 是求數(shù)組的大小。?sizeof(指針) 是求指針的大小,在 32 位機器上,任意指針都占 4 個字節(jié)的空間 。當數(shù)組作為函數(shù)的參數(shù)進行傳遞時,數(shù)組就自動退化為同類型的指針 。 ? 缺省參數(shù)的值是在編譯的時候,但確定引用、指針的虛函數(shù)調(diào)用哪個類型的函數(shù)是在運行的時候。 ? char ?p[] =? "Hello World" ;p 是一個數(shù)組,會開辟一塊內(nèi)存,并拷貝 "Hello World" 初始化該數(shù)組。
題目 :
題目(一): C++中我們可以用static 修飾一個類的成員函數(shù),也可以用 const 修飾類的成員函數(shù)(寫在函數(shù)的最后表示不能修改成員變量,不是指寫在前面表示返回值為常量)。請問:能不能同時用 static 和 const 修飾類的成員函數(shù)?
分析: 答案是不可以。 C++ 編譯器在實現(xiàn) const 的成員函數(shù)的時候為了確保該函數(shù)不能修改類的實例的狀態(tài),會在函數(shù)中添加一個隱式的參數(shù) const this* 。但當一個成員為 static 的時候,該函數(shù)是沒有 this 指針的。也就是說此時 static 的用法和 static 是沖突的。
我們也可以這樣理解:兩者的語意是矛盾的。 static 的作用是表示該函數(shù)只作用在類型的靜態(tài)變量上,與類的實例沒有關系;而 const 的作用是確保函數(shù)不能修改類的實例的狀態(tài),與類型的靜態(tài)變量沒有關系。因此不能同時用它們。
題目(二): 運行下面C++代碼,輸出是什么?
[cpp] view plain
copy class ?A??{?? };?? ??? class ?B??{?? public :??????????B()?{}?? ????????~B()?{}?? };?? ??? class ?C??{?? public :??????????C()?{}?? ????????virtual ?~C()?{}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????printf("%d,?%d,?%d/n" ,?sizeof (A),?sizeof (B),?sizeof (C));?? ????????return ?0;?? }??
分析: 答案是 1, 1, 4 。 class A 是一個空類型,它的實例不包含任何信息,本來求 sizeof 應該是 0 。但當我們聲明該類型的實例的時候,它必須在內(nèi)存中占有一定的空間,否則無法使用這些實例。至于占用多少內(nèi)存,由編譯器決定。 Visual Studio 2008 中每個空類型的實例占用一個 byte 的空間。
class B 在 class A 的基礎上添加了構(gòu)造函數(shù)和析構(gòu)函數(shù)。由于構(gòu)造函數(shù)和析構(gòu)函數(shù)的調(diào)用與類型的實例無關(調(diào)用它們只需要知道函數(shù)地址即可),在它的實例中不需要增加任何信息。所以 sizeof(B) 和 sizeof(A) 一樣,在 Visual Studio 2008 中都是 1 。
class C 在 class B 的基礎上把析構(gòu)函數(shù)標注為虛擬函數(shù)。 C++ 的編譯器一旦發(fā)現(xiàn)一個類型中有虛擬函數(shù),就會為該類型生成虛函數(shù)表,并在該類型的每一個實例中添加一個指向虛函數(shù)表的指針。在 32 位的機器上,一個指針占 4 個字節(jié)的空間,因此 sizeof(C) 是 4 。
題目(三): 運行下面的C++代碼,得到的結(jié)果是什么 ?
分析: 答案是 Print1 調(diào)用正常,打印出 hello world ,但運行至 Print2 時,程序崩潰。調(diào)用 Print1 時,并不需要 pA 的地址,因為 Print1 的函數(shù)地址是固定的。編譯器會給 Print1 傳入一個 this 指針,該指針為 NULL ,但在 Print1 中該 this 指針并沒有用到。只要程序運行時沒有訪問不該訪問的內(nèi)存就不會出錯,因此運行正常。在運行 print2 時,需要 this 指針才能得到 m_value 的值。由于此時 this 指針為 NULL ,因此程序崩潰了。
題目(四): 運行下面的C++代碼,得到的結(jié)果是什么 ?
[cpp] view plain
copy class ?A??{?? private :??????????int ?m_value;?? ??? public :??????????A(int ?value)?? ????????{?? ????????????????m_value?=?value;?? ????????}?? ????????void ?Print1()?? ????????{?? ????????????????printf("hello?world" );?? ????????}?? ????????virtual ?void ?Print2()?? ????????{?? ????????????????printf("hello?world" );?? ????????}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????A*?pA?=?NULL;?? ????????pA->Print1();?? ????????pA->Print2();?? ??? ????????return ?0;?? }??
分析: 答案是 Print1 調(diào)用正常,打印出 hello world ,但運行至 Print2 時,程序崩潰。 Print1 的調(diào)用情況和上面的題目一樣,不在贅述。由于 Print2 是虛函數(shù)。 C++ 調(diào)用虛函數(shù)的時候,要根據(jù)實例(即 this 指針指向的實例)中虛函數(shù)表指針得到虛函數(shù)表,再從虛函數(shù)表中找到函數(shù)的地址。由于這一步需要訪問實例的地址(即 this 指針),而此時 this 指針為空指針,因此導致內(nèi)存訪問出錯。
題目(五): C++中 靜態(tài)成員函數(shù)能不能同時也是虛函數(shù)?
[cpp] view plain
copy class ?A??{?? private :??????????int ?m_value;?? ??? public :??????????A(int ?value)?? ????????{?? ????????????????m_value?=?value;?? ????????}?? ????????void ?Print1()?? ????????{?? ????????????????printf("hello?world" );?? ????????}?? ????????void ?Print2()?? ????????{?? ????????????????printf("%d" ,?m_value);?? ????????}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????A*?pA?=?NULL;?? ????????pA->Print1();?? ????????pA->Print2();?? ??? ????????return ?0;?? }??
分析: 答案是不能。調(diào)用靜態(tài)成員函數(shù)不要實例。但調(diào)用虛函數(shù)需要從一個實例中指向虛函數(shù)表的指針以得到函數(shù)的地址,因此調(diào)用虛函數(shù)需要一個實例。兩者相互矛盾。
題目(六): 運行下列 C++ 代碼,輸出什么?
[cpp] view plain
copy struct ?Point3D??{?? ????????int ?x;?? ????????int ?y;?? ????????int ?z;?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????Point3D*?pPoint?=?NULL;?? ????????int ?offset?=?(int )(&(pPoint)->z);?? ??? ????????printf("%d" ,?offset);?? ????????return ?0;?? }??
答案: 輸出 8 。由于在 pPoint->z 的前面加上了取地址符號,運行到此時的時候,會在 pPoint 的指針地址上加 z 在類型 Point3D 中的偏移量 8 。由于 pPoint 的地址是 0 ,因此最終 offset 的值是 8 。
&(pPoint->z) 的語意是求 pPoint 中變量 z 的地址( pPoint 的地址 0 加 z 的偏移量 8 ),并不需要訪問 pPoint 指向的內(nèi)存。只要不訪問非法的內(nèi)存,程序就不會出錯。
題目(七): 運行下列 C++ 代碼,輸出什么?
[cpp] view plain
copy class ?A??{?? public :??????????A()?? ????????{?? ????????????????Print();?? ????????}?? ????????virtual ?void ?Print()?? ????????{?? ????????????????printf("A?is?constructed./n" );?? ????????}?? };?? ??? class ?B:?public ?A??{?? public :??????????B()?? ????????{?? ????????????????Print();?? ????????}?? ??? ????????virtual ?void ?Print()?? ????????{?? ????????????????printf("B?is?constructed./n" );?? ????????}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????A*?pA?=?new ?B();?? ????????delete ?pA;?? ??? ????????return ?0;?? }??
答案: 先后打印出兩行 :A is constructed. B is constructed.? 調(diào)用 B 的構(gòu)造函數(shù)時,先會調(diào)用 B 的基類及 A 的構(gòu)造函數(shù)。然后在 A 的構(gòu)造函數(shù)里調(diào)用 Print 。由于此時實例的類型 B 的部分還沒有構(gòu)造好,本質(zhì)上它只是 A 的一個實例,他的虛函數(shù)表指針指向的是類型 A 的虛函數(shù)表。因此此時調(diào)用的 Print 是 A::Print ,而不是 B::Print 。接著調(diào)用類型 B 的構(gòu)造函數(shù),并調(diào)用 Print 。此時已經(jīng)開始構(gòu)造 B ,因此此時調(diào)用的 Print 是 B::Print 。
同樣是調(diào)用虛擬函數(shù) Print ,我們發(fā)現(xiàn)在類型 A 的構(gòu)造函數(shù)中,調(diào)用的是 A::Print ,在 B 的構(gòu)造函數(shù)中,調(diào)用的是 B::Print 。因此虛函數(shù)在構(gòu)造函數(shù)中,已經(jīng)失去了虛函數(shù)的動態(tài)綁定特性。
題目(八): 在 C++ 中, struct 和 class 有什么不同 ?
答案: 在 C++ 中,如果沒有標明函數(shù)或者變量是的訪問權(quán)限級別,在 struct 中,是 public 的;而在 class 中,是 private 的。
題目( 九) :運行下圖中的 C++代碼,輸出是什么 ?
[cpp] view plain
copy #include?<iostream> ????? class ?A??{?? private :??????????int ?n1;?? ????????int ?n2;?? public :??????????A():?n2(0),?n1(n2?+?2)?? ????????{?? ????????}?? ??? ????????void ?Print()?? ????????{?? ????????????????std::cout?<<?"n1:?" ?<<?n1?<<?",?n2:?" ?<<?n2?<<?std::endl;?? ????????}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????A?a;?? ????????a.Print();?? ??? ????????return ?0;?? }??
答案 :輸出 n1是一個隨機的數(shù)字, n2為 0。在 C++中,成員變量的初始化順序與變量在類型中的申明順序相同,而與它們在構(gòu)造函數(shù)的初始化列表中的順序無關。因此在這道題中,會首先初始化 n1,而初始 n1的參數(shù) n2還沒有初始化,是一個隨機值,因此 n1就是一個隨機值。初始化 n2時,根據(jù)參數(shù) 0對其初始化,故 n2=0。
題目( 十) : 編譯運行下圖中的 C++代碼,結(jié)果是什么?( A)編譯錯誤;( B)編譯成功,運行時程序崩潰;( C)編譯運行正常,輸出 10。請選擇正確答案并分析原因 。
[cpp] view plain
copy #include?<iostream> ????? class ?A??{?? private :??????????int ?value;?? ??? public :??????????A(int ?n)?? ????????{?? ????????????????value?=?n;?? ????????}?? ??? ????????A(A?other)?? ????????{?? ????????????????value?=?other.value;?? ????????}?? ??? ????????void ?Print()?? ????????{?? ????????????????std::cout?<<?value?<<?std::endl;?? ????????}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????A?a?=?10;?? ????????A?b?=?a;?? ????????b.Print();?? ??? ????????return ?0;?? }??
答案 :編譯錯誤。在復制構(gòu)造函數(shù)中傳入的參數(shù)是 A的一 個實例。由于是傳值,把形參拷貝到實參會調(diào)用復制構(gòu)造函數(shù)。因此如果允許復制構(gòu)造函數(shù)傳值,那么會形成永無休止的遞歸并造成棧溢出。因此 C++的標準不允許復制構(gòu)造函數(shù)傳值參數(shù),而必須是傳引用或者常量引用。在 Visual Studio和 GCC中,都將編譯出 錯。
題目( 十一) : 運行下圖中的 C++代碼,輸出是什么 ?
[cpp] view plain
copy int ?SizeOf(char ?pString[])??{?? ????????return ?sizeof (pString);?? }?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????char *?pString1?=?"google" ;?? ????????int ?size1?=?sizeof (pString1);?? ????????int ?size2?=?sizeof (*pString1);?? ??? ????????char ?pString2[100]?=?"google" ;?? ????????int ?size3?=?sizeof (pString2);?? ????????int ?size4?=?SizeOf(pString2);?? ??? ????????printf("%d,?%d,?%d,?%d" ,?size1,?size2,?size3,?size4);?? ??? ????????return ?0;?? }??
答案 : 4, 1, 100, 4。 pString1是一個指針。在 32位機器上,任意指針都占 4個字節(jié)的空間。 *pString1是字符串 pString1的第一個字符。一個字符占一個字節(jié)。 pString2是一個數(shù)組, sizeof(pString2)是求數(shù)組的大小。這個數(shù)組包含 100個字符,因此大小是 100個字節(jié)。而在函數(shù) SizeOf中,雖然傳入的參數(shù)是一個字符數(shù)組, 當數(shù)組作為函數(shù)的參數(shù)進行傳遞時,數(shù)組就自動退化為同類型的指針。
題目( 十二) :運行下圖中代碼,輸出的結(jié)果是什么?這段代碼有什么問題 ?
[cpp] view plain
copy #include?<iostream> ????? class ?A??{?? public :??????????A()?? ????????{???????std::cout?<<?"A?is?created." ?<<?std::endl;????????}?? ??? ????????~A()?? ????????{???????std::cout?<<?"A?is?deleted." ?<<?std::endl;????????}?? };?? ??? class ?B?:?public ?A??{?? public :??????????B()?? ????????{???????std::cout?<<?"B?is?created." ?<<?std::endl;????????}?? ??? ????????~B()?? ????????{???????std::cout?<<?"B?is?deleted." ?<<?std::endl;????????}?? };?? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ????????A*?pA?=?new ?B();?? ????????delete ?pA;?? ??? ????????return ?0;?? }??
答案 :輸出三行,分別是: A is created. B is created. A is deleted。用 new創(chuàng)建 B時,回調(diào)用 B的構(gòu)造函數(shù)。在調(diào)用 B的構(gòu)造函數(shù)的時候,會先調(diào)用 A的構(gòu)造函數(shù)。因此先輸出A is created. B is created.
接下來運行 delete語句時,會調(diào)用析構(gòu)函數(shù)。由于 pA被聲明成類型 A的指針,同時基類 A的析構(gòu)函數(shù)沒有標上 virtual,因此只有 A的析構(gòu)函數(shù)被調(diào)用到,而不會調(diào)用 B的析構(gòu)函數(shù)。
由于 pA實際上是指向一個 B的實例的指針,但在析構(gòu)的時候只調(diào)用了基類 A的析構(gòu)函數(shù),卻沒有調(diào)用 B的析構(gòu)函數(shù)。這就是一個問題。如果在類型 B中創(chuàng)建了一些資源,比如文件句柄、內(nèi)存等,在這種情況下都得不到釋放,從而導致資源泄漏。
問題(十三) :運行如下的C++代碼,輸出是什么?
[cpp] view plain
copy class ?A??{?? public :?????virtual ?void ?Fun(int ?number?=?10)?? ???{?? ???????std::cout?<<?"A::Fun?with?number?" ?<<?number;?? ???}?? };?? ??? class ?B:public ?A??{?? public :?????virtual ?void ?Fun(int ?number?=?20)?? ???{?? ???????std::cout?<<?"B::Fun?with?number?" ?<<?number;?? ???}?? };?? ??? int ?main()??{?? ???B?b;?? ???A?&a?=?b;?? ???a.Fun();?? }??
答案 :輸出B::Fun with number 10。由于a是一個指向B實例的引用,因此在運行的時候會調(diào)用B::Fun。但缺省參數(shù)是在編譯期決定的。在編譯的時候,編譯器只知道a是一個類型a的引用,具體指向什么類型在編譯期是不能確定的,因此會按照A::Fun的聲明把缺省參數(shù)number設為10。
???????????這一題的關鍵在于理解確定缺省參數(shù)的值是在編譯的時候,但確定引用、指針的虛函數(shù)調(diào)用哪個類型的函數(shù)是在運行的時候。
問題(十四) :運行如下的C代碼,輸出是什么?
[cpp] view plain
copy char *?GetString1()??{?? ???char ?p[]?=?"Hello?World" ;?? ???return ?p;?? }?? ??? char *?GetString2()??{?? ???char ?*p?=?"Hello?World" ;?? ???return ?p;?? }?? ??? ??? int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ???printf("GetString1?returns:?%s.?/n" ,?GetString1());?? ???printf("GetString2?returns:?%s.?/n" ,?GetString2());?? ??? ???return ?0;?? }??
答案 :輸出兩行,第一行GetString1 returns:?后面跟的是一串隨機的內(nèi)容,而第二行GetString2 returns: Hello World.兩個函數(shù)的區(qū)別在于GetString1中是一個數(shù)組,而GetString2中是一個指針。
當運行到GetString1時,p是一個數(shù)組,會開辟一塊內(nèi)存,并拷貝"Hello World"初始化該數(shù)組。接著返回數(shù)組的首地址并退出該函數(shù)。由于p是GetString1內(nèi)的一個局部變量,當運行到這個函數(shù)外面的時候,這個數(shù)組的內(nèi)存會被釋放掉。因此在_tmain函數(shù)里再去訪問這個數(shù)組的內(nèi)容時,結(jié)果是隨機的。
當運行到GetString2時,p是一個指針,它指向的是字符串常量區(qū)的一個常量字符串。該常量字符串是一個全局的,并不會因為退出函數(shù)GetString2而被釋放掉。因此在_tmain中仍然根據(jù)GetString2返回的地址得到字符串"Hello World"。
問題(十五) :運行下圖中C代碼,輸出的結(jié)果是什么?
[cpp] view plain
copy int ?_tmain(int ?argc,?_TCHAR*?argv[])??{?? ???char ?str1[]?=?"hello?world" ;?? ???char ?str2[]?=?"hello?world" ;?? ??? ???char *?str3?=?"hello?world" ;?? ???char *?str4?=?"hello?world" ;?? ??? ???if (str1?==?str2)?? ???????printf("str1?and?str2?are?same./n" );?? ???else ?? ???????printf("str1?and?str2?are?not?same./n" );?? ??? ???if (str3?==?str4)?? ???????printf("str3?and?str4?are?same./n" );?? ???else ?? ???????printf("str3?and?str4?are?not?same./n" );?? ??? ???return ?0;?? }??
答案 : 輸出兩行。第一行是str1 and str2 are not same,第二行是str3 and str4 are same。
str1和str2是兩個字符串數(shù)組。我們會為它們分配兩個長度為12個字節(jié)的空間,并把"hello world"的內(nèi)容分別拷貝到數(shù)組中去。這是兩個初始地址不同的數(shù)組,因此比較str1和str2的值,會不相同。str3和str4是兩個指針,我們無需為它們分配內(nèi)存以存儲字符串的內(nèi)容,而只需要把它們指向"hello world“在內(nèi)存中的地址就可以了。由于"hello world”是常量字符串,它在內(nèi)存中只有一個拷貝,因此str3和str4指向的是同一個地址。因此比較str3和str4的值,會是相同的。
問題(十六) :運行下圖中的C++代碼,打印出的結(jié)果是什么?
[cpp] ?view plaincopy
bool ?Fun1( char *?str)?? {?? ????printf("%s\n" ,str);?? ????return ? false ;?? }?? ?? bool ?Fun2( char *?str)?? {?? ????printf("%s\n" ,str);?? ????return ? true ;?? }?? ?? int ?_tmain( int ?argc,?_TCHAR*?argv[])?? {?? ????bool ?res1,res2;?? ????res1?=?(Fun1("a" )&&?Fun2( "b" ))?||?(Fun1( "c" )?||?Fun2( "d" ));?? ????res2?=?(Fun1("a" )&&?Fun2( "b" ))?&&(Fun1( "c" )?||?Fun2( "d" ));?? ?? ????return ?res1||?res2;?? }??
答案
:打印出4行,分別是a、c、d、a。
在C/C++中,與、或運算是從左到右的順序執(zhí)行的。在計算rest1時,先計算Fun1(“a”)&& Func2(“b”)。首先Func1(“a”)打印出內(nèi)容為a的一行。由于Fun1(“a”)返回的是false,無論Func2(“b”)的返回值是true還是false,Fun1(“a”)&& Func2(“b”)的結(jié)果都是false。由于Func2(“b”)的結(jié)果無關重要,因此Func2(“b”)會略去而不做計算。接下來計算Fun1(“c”)|| Func2(“d”),分別打印出內(nèi)容c和d的兩行。
??????????????? 在計算rest2時,首先Func1(“a”)打印出內(nèi)容為a的一行。由于Func1(“a”)返回false,和前面一樣的道理,Func2(“b”)會略去不做計算。由于Fun1(“a”)&& Func2(“b”)的結(jié)果是false,不管Fun1(“c”)&& Func2(“d”)的結(jié)果是什么,整個表達式得到的結(jié)果都是false,因此Fun1(“c”) || Func2(“d”)都將被忽略。
問題(十七) :運行下面的C++代碼,打印的結(jié)果是什么?
[cpp] ?view plaincopy
class ?Base?? {?? public :?? ????voidprint()?{?doPrint();}?? ?? private :?? ????virtual ? void ?doPrint()?{cout?<<? "Base::doPrint" ?<<?endl;}?? };?? ?? class ?Derived?:? public ?Base?? {?? private :?? ????virtual ? void ?doPrint()?{cout?<<? "Derived::doPrint" ?<<?endl;}?? };?? ?? int ?_tmain( int ?argc,?_TCHAR*?argv[])?? {?? ????Base?b;?? ????b.print();?? ?? ????Derived?d;?? ????d.print();?? ?? ????return ?0;?? }??
答案 :輸出兩行,分別是Base::doPrint和Derived::doPrint。在print中調(diào)用doPrint時,doPrint()的寫法和this->doPrint()是等價的,因此將根據(jù)實際的類型調(diào)用對應的doPrint。所以結(jié)果是分別調(diào)用的是Base::doPrint和Derived::doPrint2。如果感興趣,可以查看一下匯編代碼,就能看出來調(diào)用doPrint是從虛函數(shù)表中得到函數(shù)地址的。
總結(jié)
以上是生活随笔 為你收集整理的C/C++面试题精选 的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
如果覺得生活随笔 網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔 推薦給好友。