C の仮想継承を理解する
オブジェクト指向プログラミングでは、仮想継承は多重継承が引き起こすダイヤモンド問題を解決するメカニズムです。オブジェクトの初期化における曖昧さのため。次のコード スニペットを考えてみましょう:
class Base { public: Base(Base* pParent); /* implements basic stuff */ }; class A : virtual public Base { public: A(A* pParent) : Base(pParent) {} /* ... */ }; class B : virtual public Base { public: B(B* pParent) : Base(pParent) {} /* ... */ }; class C : public A, public B { public: C(C* pParent) : A(pParent), B(pParent) {} // - Compilation error here /* ... */ };
このコードをコンパイルしようとすると、GCC はマークされた行でコンパイル エラーを報告します。このエラーの理由は、仮想継承が通常の継承とどのように異なるかにあります。
仮想継承では、最も派生したクラスと仮想基本クラスの間に間接的な関係が導入されます。この場合、C は Base から直接継承するのではなく、その仮想基底クラス A および B を通じて継承します。これは、C のコンストラクター内の Base の初期化が、最も派生したクラスのコンストラクター (この場合は C) に委任されることを意味します。 .
ただし、C は初期化子リストで Base サブオブジェクトを明示的に初期化しません。したがって、このサブオブジェクトを初期化するには、Base のデフォルトのコンストラクターを呼び出す必要があります。ただし、Base のデフォルト コンストラクターは、C の直接の基本クラスではないため、C のコンストラクターのスコープ内ではアクセスできません。
この問題を解決するには、C は、Base のデフォルト コンストラクターを明示的に呼び出す必要があります。イニシャライザリスト。これにより、仮想 Base サブオブジェクトが適切に初期化されることが保証されます。修正されたコードは次のようになります:
class C : public A, public B { public: C(C* pParent) : A(pParent), B(pParent), Base() {} // - Explicit call to Base() /* ... */ };
以上が仮想継承は C のダイヤモンド問題をどのように解決しますか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。