日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

C++虚继承(八) --- 虚继承与继承的差异

發布時間:2024/4/11 c/c++ 30 豆豆
生活随笔 收集整理的這篇文章主要介紹了 C++虚继承(八) --- 虚继承与继承的差异 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前面一篇文章,說明了在C++ 虛繼承對基類構造函數調用順序的影響。經過仔細推敲,發現沒有徹底說清楚虛繼承與普通繼承之間的關系。所以用下面的文字再說明一下。

首先,重復一下虛擬繼承與普通繼承的區別有:

假設derived 繼承自base類,那么derived與base是一種“is a”的關系,即derived類是base類,而反之錯誤;

假設derived 虛繼承自base類,那么derivd與base是一種“has a”的關系,即derived類有一個指向base類的vptr。

因此虛繼承可以認為不是一種繼承關系,而可以認為是一種組合的關系。因為虛繼承有著“繼承”兩個關鍵字,那么大部分人都認為虛繼承與普通繼承的用法沒什么太大的不同。由此用在繼承體系中,這種將虛繼承認為是普通繼承的危害更佳大。下面先用一個例子來說明問題:

[cpp]?view plaincopy
  • class?base??
  • {??
  • public:??
  • ????base(){cout<<"base::base()!"<<endl;}??
  • ????void?printBase(){cout<<"base::printBase()!"<<endl;}??
  • };??
  • ??
  • class?derived:public?base??
  • {??
  • public:??
  • ????derived(){cout<<"derived::derived()!"<<endl;}??
  • ????void?printDerived(){cout<<"derived::printDerived()!"<<endl;}??
  • };??

  • 上面是普通繼承實現,在實際應用中,我們可以使用下面的代碼進行類型轉換:

    [cpp]?view plaincopy
  • int?main(int?argc,?const?char?*?argv[])??
  • {??
  • ????derived?oo;??
  • ????base?oo1(static_cast<base>(oo));??
  • ????oo1.printBase();??
  • ????derived?oo2?=?static_cast<derived&>(oo1);??
  • ????oo2.printDerived();??
  • ????return?0;??
  • }??
  • 編譯無錯誤,而且會得出正確的結果。其結果為:
    ?base::base()!
    ?derived::derived()!
    ?base::printBase()!
    ?derived::printDerived()!

    而將上面的普通繼承變成虛擬繼承,如下代碼:

    [cpp]?view plaincopy
  • class?base1??
  • {??
  • public:??
  • ????base1(){cout<<"base::base()!"<<endl;}??
  • ????void?printBase(){cout<<"base::printBase()!"<<endl;}??
  • };??
  • ??
  • class?derived1:virtual?public?base1??
  • {??
  • public:??
  • ????derived1(){cout<<"derived::derived()!"<<endl;}??
  • ????void?printDerived(){cout<<"derived::printDerived()!"<<endl;}??
  • };??
  • ??
  • int?main(int?argc,?const?char?*?argv[])??
  • {??
  • ????derived1?oo;??
  • ????base1?oo1(static_cast<base1>(oo));??
  • ????oo1.printBase();??
  • ????derived1?oo2?=?static_cast<derived1&>(oo1);??
  • ????oo2.printDerived();??
  • ????return?0;??
  • }??

  • 編譯上面的代碼,提示如下:


    可以看到不能將基類通過static_cast轉換為繼承類。我們知道c++提供的強制轉換函數static_cast對于繼承體系中的類對象的轉換一般是可行的。那么這里為什么就不可以了呢?還是需要從虛擬繼承的內部實現來說明問題。

    virtual base class的原始模型是在class object中為每一個有關聯的virtual base class加上一個指針vptr,該指針指向virtual基類表。有的編譯器是在繼承類已存在的virtual table直接擴充導入一個virtual base class table。不管怎么樣由于虛繼承已完全破壞了繼承體系,不能按照平常的繼承體系來進行類型轉換。

    不管怎么樣,虛繼承在類型轉換是一定要十分注意。不要輕易使用虛繼承,更不要在虛繼承的基礎上進行類型轉換,切記切記!

    總結

    以上是生活随笔為你收集整理的C++虚继承(八) --- 虚继承与继承的差异的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。