C++ Virtual關(guān)鍵字的具體使用
基礎(chǔ)理解和demo
普通的繼承
#include<iostream>
class Parent {
public:
void print() {
std::cout << "Parent" << std::endl;
}
};
class Child : Parent {
public:
void print() {
std::cout << "Child" << std::endl;
}
};
int main() {
Child c;
c.print();
return 0;
}
輸出結(jié)果為 "Child"
但是如果是 "父類的指針指向子類的對象" 這種情況下, 使用這個父類的指針去調(diào)用被重寫的方法呢, 結(jié)果會是什么呢? 從語法的本質(zhì)上講, 子類對象的內(nèi)存前半部分就是父類, 因為可以將子類對象的指針直接轉(zhuǎn)化為父類。
#include<iostream>
class Parent {
public:
void print() {
std::cout << "Parent" << std::endl;
}
};
// 注意這里必須是 public Parent
// 不然會報錯 cannot cast 'Child' to its private base class 'Parent'
class Child : public Parent {
public:
void print() {
std::cout << "Child" << std::endl;
}
};
int main() {
Parent* p = new Child();
p->print();
return 0;
}
這個時候輸出的是 "Parent" 。
所以, 當(dāng)一個成員函數(shù)需要被子類重寫, 那么必須將其聲明為virtual, 也就是 虛函數(shù) , 注意, 子類覆寫的方法的virtual關(guān)鍵字會自動繼承而來, 可以顯示地寫或者不寫(建議還是寫上)。
這樣修改完就沒問題了:
#include<iostream>
class Parent {
public:
virtual void print() {
std::cout << "Parent" << std::endl;
}
};
class Child : public Parent {
public:
virtual void print() {
std::cout << "Child" << std::endl;
}
};
int main() {
Parent* p = new Child();
p->print();
return 0;
}
加深理解
Virtual 關(guān)鍵字的一個重要概念 "只有在通過基類指針或引用間接指向派生類子類型時多態(tài)性才會起作用" , 也就是說, 基類的函數(shù)調(diào)用如果有virtual則根據(jù)多態(tài)性調(diào)用派生類的,如果沒有virtual則是正常的靜態(tài)函數(shù)調(diào)用,還是調(diào)用基類的。
舉個例子
#include <iostream.h>
class Base
{
public:
virtual void f(float x){ cout << "Base::f(float) " << x << endl; }
void g(float x){ cout << "Base::g(float) " << x << endl; }
void h(float x){ cout << "Base::h(float) " << x << endl; }
};
class Derived : public Base
{
public:
virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }
void g(int x){ cout << "Derived::g(int) " << x << endl; }
void h(float x){ cout << "Derived::h(float) " << x << endl; }
};
void main(void)
{
Derived d;
Base *pb = &d;
Derived *pd = &d;
// Good : behavior depends solely on type of the object
pb->f(3.14f); // Derived::f(float) 3.14
pd->f(3.14f); // Derived::f(float) 3.14
// Bad : behavior depends on type of the pointer
pb->g(3.14f); // Base::g(float) 3.14
pd->g(3.14f); // Derived::g(int) 3 (surprise!)
// Bad : behavior depends on type of the pointer
pb->h(3.14f); // Base::h(float) 3.14 (surprise!)
pd->h(3.14f); // Derived::h(float) 3.14
}
粘貼這個博客的一段話, 表達(dá)的就是這個意思:
bp 和dp 指向同一地址,按理說運行結(jié)果應(yīng)該是相同的,而事實上運行結(jié)果不同,所以他把原因歸結(jié)為C++的隱藏規(guī)則,其實這一觀點是錯的。決定bp和dp調(diào)用函數(shù)運行結(jié)果的不是他們指向的地址,而是他們的指針類型。 “只有在通過基類指針或引用間接指向派生類子類型時多態(tài)性才會起作用”(C++ Primer 3rd Edition)。pb是基類指針,pd是派生類指針,pd的所有函數(shù)調(diào)用都只是調(diào)用自己的函數(shù),和多態(tài)性無關(guān),所以pd的所有函數(shù)調(diào)用的結(jié)果都輸出Derived::是完全正常的;pb的函數(shù)調(diào)用如果有virtual則根據(jù)多態(tài)性調(diào)用派生類的,如果沒有virtual則是正常的靜態(tài)函數(shù)調(diào)用,還是調(diào)用基類的,所以有virtual的f函數(shù)調(diào)用輸出Derived::,其它兩個沒有virtual則還是輸出Base::很正常啊 ,nothing surprise! 所以并沒有所謂的隱藏規(guī)則,雖然《高質(zhì)量C++/C 編程指南》是本很不錯的書,可大家不要迷信哦。記住“只有在通過基類指針或引用間接指向派生類子類型時多態(tài)性才會起作用”。
到此這篇關(guān)于C++ Virtual關(guān)鍵字的具體使用的文章就介紹到這了,更多相關(guān)C++ Virtual關(guān)鍵字內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++運行時獲取類型信息的type_info類與bad_typeid異常
- 詳解C++編程中的sizeof運算符與typeid運算符
- 一文讀懂c++之static關(guān)鍵字
- C/C++ 中const關(guān)鍵字的用法小結(jié)
- 詳解c++中的 static 關(guān)鍵字及作用
- C++面試基礎(chǔ)之static關(guān)鍵字詳解
- C++11關(guān)于auto關(guān)鍵字的使用示例
- c++使用正則表達(dá)式提取關(guān)鍵字的方法
- C++中volatile關(guān)鍵字及常見的誤解總結(jié)
- C++ override關(guān)鍵字使用詳解
- c++ typeid關(guān)鍵字的使用
相關(guān)文章
C++實現(xiàn)LeetCode(138.拷貝帶有隨機指針的鏈表)
這篇文章主要介紹了C++實現(xiàn)LeetCode(138.拷貝帶有隨機指針的鏈表),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
C語言中不定參數(shù)?...?的語法以及函數(shù)封裝
不定參數(shù)是指函數(shù)可以接收不確定個數(shù)的參數(shù),下面這篇文章主要給大家介紹了關(guān)于C語言中不定參數(shù)?...?的語法以及函數(shù)封裝的相關(guān)資料,文中通過實例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-01-01
c++實現(xiàn)獲取當(dāng)前時間(精確至秒,毫秒和微妙)
這篇文章主要為大家詳細(xì)介紹了c++實現(xiàn)獲取當(dāng)前時間(可以精確至秒,毫秒和微妙)的相關(guān)知識,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以參考一下2023-11-11
C++ 中 const和static readonly區(qū)別
這篇文章主要介紹了C++ 中 const和static readonly區(qū)別的相關(guān)資料,需要的朋友可以參考下2017-05-05
C++?反匯編之關(guān)于Switch語句的優(yōu)化措施
這篇文章主要介紹了C++?反匯編之關(guān)于Switch語句的優(yōu)化措施,利用三種優(yōu)化來降低樹高度,誰的效率高就優(yōu)先使用誰,三種優(yōu)化都無法匹配才會使用判定樹,具體內(nèi)容詳情跟隨小編一起看看吧2022-01-01

