C++學習---- virtual的三種用法


virtual用法一:多態

 1 #include<iostream> 
 2 using namespace std;
 3 class A{
 4 public:
 5 virtual void display(){ cout<<"A"<<endl; }
 6 };
 7 class B : public A{
 8 public:
 9 void display(){ cout<<"B"<<endl; }
10 };
11 void doDisplay(A *p)
12 {
13     p->display();
14     delete p;
15 }
16 
17 int main(int argc,char* argv[])
18 {
19     doDisplay(new B());
20     return 0;
21 }

      這段代碼打印出的結果為B,但是當把A類中的virtual去掉之后打印出的就為A。當基類中沒有virtual的時候,編譯器在編譯的時候把p看做A類的對象,調用的自然就是A類的方法。但是加上virtual之后,將dispaly方法變成了虛方法,這樣調用的時候編譯器會看調用的究竟是誰的實例化對象,這樣就實現了多態的效果。也就是說,當基類的派生類中有重寫過基類的虛方法的時候,使用基類的指針指向派生類的對象,調用這個方法實際上調用的會是派生類最后實現的方法。

virtual用法二: 虛繼承:解決數據冗余

 1 #include<iostream> 
 2 using namespace std;
 3 class Person{
 4 public: Person(){ cout<<"Person構造"<<endl; }
 5 ~Person(){ cout<<"Person析構"<<endl; }
 6 };
 7 class Teacher : virtual public Person{
 8 public: Teacher(){ cout<<"Teacher構造"<<endl; }
 9 ~Teacher(){ out<<"Teacher析構"<<endl; }
10 };
11 class Student : virtual public Person{
12 public: Student(){ cout<<"Student構造"<<endl; }
13 ~Student(){ cout<<"Student析構"<<endl; }
14 };
15 class TS : public Teacher, public Student{
16 public: TS(){ cout<<"TS構造"<<endl; }
17 ~TS(){ cout<<"TS析構"<<endl; }
18 };
19 int main(int argc,char* argv[])
20 {
21     TS ts;
22     return 0;
23 }

這段代碼的終端輸出結果為: 

1 Person構造 
2 Teacher構造 
3 Student構造 
4 TS構造 
5 TS析構 
6 Student析構 
7 Teacher析構 
8 Person析構

當Teacher類和Student類沒有虛繼承Person類的時候,也就是把virtual去掉時候終端輸出的結果為: 

 1 Person構造
 2 Teacher構造
 3 Person構造
 4 Student構造
 5 TS構造
 6 TS析構
 7 Student析構
 8 Person析構
 9 Teacher析構
10 Person析構

大家可以很清楚的看到這個結果明顯不是我們所期望的。我們在構造TS的時候需要先構造他的基類,也就是Teacher類和Student類。而Teacher類和Student類由都繼承於Person類。這樣就導致了構造TS的時候實例化了兩個Person類。同樣的道理,析構的時候也是析構了兩次Person類,這是非常危險的,也就引發出了virtual的第三種用法,虛析構。

virtual用法三:虛析構:可以防止內存泄漏

 1 #include<iostream>
 2 using namespace std;
 3 class Person{
 4 public: Person() {name = new char[16];cout<<"Person構造"<<endl;}
 5 virtual ~Person() {delete []name;cout<<"Person析構"<<endl;}
 6 private:
 7 char *name;
 8 };
 9 class Teacher :virtual public Person{
10 public: Teacher(){ cout<<"Teacher構造"<<endl; }
11 ~Teacher(){ cout<<"Teacher析構"<<endl; }
12 };
13 class Student :virtual public Person{
14 public: Student(){ cout<<"Student構造"<<endl; }
15 ~Student(){ cout<<"Student析構"<<endl; }
16 };
17 class TS : public Teacher,public Student{
18 public: TS(){ cout<<"TS構造"<<endl; }
19 ~TS(){ cout<<"TS析構"<<ENDL; }
20 };
21 int main(int argc,char* argv[])
22 {
23 Person *p = new TS();
24 delete p;
25 return 0;
26 }

這段代碼的運行結果為:

1 Person構造
2 Teacher構造
3 Student構造
4 TS構造
5 TS析構
6 Student析構
7 Teacher析構
8 Person析構

但是當我們把Person類中析構前面的virtual去掉之后的運行結果為:

1 Person構造
2 Teacher構造
3 Student構造
4 TS構造
5 Person析構
6 程序崩潰

很明顯這個結果不是我們想要的程序,崩潰造成的后果是不可預計的,所以我們一定要注意在基類的析構函數前面加上virtual,使其變成虛析構在C++程序中使用虛函數,虛繼承和虛析構是很好的習慣 可以避免許多的問題。

原文:https://blog.csdn.net/jirryzhang/article/details/79392934


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM