深入理解c++中virtual關鍵字_C 語言

來源:互聯網
上載者:User

1.virtual關鍵字主要是什麼作用?
c++中的函數調用預設不適用動態綁定。要觸發動態綁定,必須滿足兩個條件:第一,指定為虛函數;第二,通過基類類型的引用或指標調用。
由此可見,virtual主要主要是實現動態綁定。

2.那些情況下可以使用virtual關鍵字?
virtual可用來定義類函數和應用到虛繼承。

友元函數 建構函式 static靜態函數 不能用virtual關鍵字修飾;
普通成員函數 和解構函式 可以用virtual關鍵字修飾;

3.virtual函數的效果

複製代碼 代碼如下:

class GrandFather
{
public:
 GrandFather() {}
 virtual void fun()
 {
  cout << "GrandFather call function!" << endl;
 }
};

class Father : public GrandFather
{
public:
  Father() {}
  void fun()
  {
   cout << "Father call function!" << endl;
  }
};


class Son : public Father
{
public:
 Son() {}
 void fun()
 {
  cout << "Son call function!" << endl;
 }
};

void print(GrandFather* father)
{
 father->fun();
}

int _tmain(int argc, _TCHAR* argv[])
{
 Father * pfather = new Son;
        pfather->fun();
        GrandFather * pgfather = new Father;
        print(pgfather);
 return 0;
}


輸出為 Son call function
       Father call function

4.virtual的繼承性
只要基函數定義了virtual,繼承類的該函數也就具有virtual屬性
即 GrandFather Father Son同時定義virtual void fun()與GrandFather一個定義virtual void fun效果是一樣的

5.虛解構函式

複製代碼 代碼如下:

class GrandFather
{
public:
 GrandFather() {}
 virtual void fun()
 {
  cout << "GrandFather call function!" << endl;
 }

 ~GrandFather()
 {
  cout << "GrandFather destruction!" << endl;
 }
};

class Father : public GrandFather
{
public:
 Father() {}
 void fun()
 {
  cout << "Father call function!" << endl;
 }

 ~Father()
 {
  cout << "Father destruction!" << endl;
 }
};


class Son : public Father
{
public:
 Son() {}
 void fun()
 {
  cout << "Son call function!" << endl;
 }

  ~Son()
 {
  cout << "Son destruction!" << endl;
 }
};

void print(GrandFather* p)
{
 p->fun();
}

int _tmain(int argc, _TCHAR* argv[])
{
 Father * pfather = new Son;
 delete pfather;
 return 0;
}


以上代碼輸出:Father destruction!
                             GrandFather destruction!
執行了Son的建構函式,沒執行Son的解構函式,故把GrandFather的解構函式設定為virtual
則輸出: Son destruction!
        Father Destruction!
        GrandFather destruction!

6. 純虛函數
純虛函數定義如下:

複製代碼 代碼如下:

class GrandFather
{
public:
 GrandFather() {}
 virtual void fun() = 0
 {
  cout << "GrandFather call function!" << endl;
 }

 virtual ~GrandFather()
 {
  cout << "GrandFather destruction!" << endl;
 }
};


純虛函數為後代類提供可覆蓋的介面,但這個類中的版本決不會調用。
含有(或繼續)一個或多個純虛函數的類是抽象基類,抽象基類不能執行個體化!
繼承類只有重寫這個介面才能被執行個體化


7.虛繼承
虛繼承主要解決交叉繼承帶來的問題。這裡給出一片參考文章c++虛繼承。
給一個例子如下

複製代碼 代碼如下:

class GrandFather
{
public:
 GrandFather() {}
 void fun()
 {
  cout << "GrandFather call function!" << endl;
 }

 virtual ~GrandFather()
 {
  cout << "GrandFather destruction!" << endl;
 }
};

class Father1 : public GrandFather
{
public:
 Father1() {}
 void fun()
 {
  cout << "Father call function!" << endl;
 }

};

class Father2 : public GrandFather
{
public:
 Father2() {}
 void fun()
 {
  cout << "Father call function!" << endl;
 }

};


class Son : public Father1, public Father2
{
public:
 Son() {}
 //void fun()
 //{
 // cout << "Son call function!" << endl;
 //}
};

void print(GrandFather* p)
{
 p->fun();
}

int _tmain(int argc, _TCHAR* argv[])
{
 Son* son = new Son;
 son->fun();
 return 0;
}


編譯時間會提示報錯對fun的訪問不明確
如果Father1和Father2都用虛繼承繼承GrandFather類則可以解決這個問題

8. 建構函式和解構函式中的虛函數
如果在建構函式或解構函式中調用虛函數,則啟動並執行是為建構函式或解構函式自身類型定義的版本


9.虛函數的實現機制
關於虛函數的實現機制,我們以後在介紹。

10.小結
關於virtual關鍵字的用法總結如上,有錯誤或者總結不到位的情況請能幫本人指出!

11.例子

複製代碼 代碼如下:

class classA
{
 public:
 classA()
 {
  clear();
 }
 virtual ~classA()
 {
 }
 void clear()
 {
  memset(this , 0 , sizeof(*this));
 }
 virtual void func()
 {
  printf("func\n");
 }
};

class classB : public classA
{
};

int main(void)
{
 classA oa;
 classB ob;
 classA * pa0 = &oa;
 classA * pa1 = &ob;
 classB * pb = &ob;
 oa.func(); // 1
 ob.func(); // 2
 pa0->func(); // 3
 pa1->func(); // 4
 pb->func(); // 5
 return 0;
}


補充一個例子,這個程式輸出依次是
func
func
出錯
func
func

談談我的理解,當
classA oa;
oa.func();
不存在動態調用的過程,所以func雖然是虛函數,但是函數調用不通過虛表訪問,所以即使

複製代碼 代碼如下:

memset(this , 0 , sizeof(*this));

找不到虛表地址也沒有關係
在執行classB ob;的時候,注意memset的是classA的地址,所有ob的虛表是存在的
即是如下,通過指標或引用(動態綁定)訪問oa的func函數(需要從虛表訪問),會出錯
訪問ob的func和函數,無論靜態訪問還是動態訪問,都不會出錯


當把classB的代碼改成如下時

複製代碼 代碼如下:

class classB : public classA

<PRE style="FONT-WEIGHT: bold" class=cpp name="code">{</PRE><PRE style="FONT-WEIGHT: bold" class=cpp name="code">        classB()
 {
  clear();
 }
 virtual ~classB()
 {
 }
 void clear()
 {
     memset(this , 0 , sizeof(*this));
 }</PRE><BR>
<PRE></PRE>
<PRE style="FONT-WEIGHT: bold" class=cpp name="code">};</PRE>輸出為


func
func
出錯
出錯
出錯

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.