首頁 >後端開發 >C++ >呼叫多型方法時如何解出型別重疊的多重繼承中的二義性?

呼叫多型方法時如何解出型別重疊的多重繼承中的二義性?

Linda Hamilton
Linda Hamilton原創
2024-10-25 06:23:02344瀏覽

How to Resolve Ambiguity in Multiple Inheritance with Overlapping Types When Calling Polymorphic Methods?

重疊類型的多重繼承中的歧義消除

當不同的基類具有不重疊的集時,多重繼承可能會導致不明確的類別成員呼叫多型方法的適用型別。考慮這樣的場景:可變參數基底類別模板 Base 定義了一個方法 foo(),該方法只能使用其類型參數包中包含的模板參數來呼叫。

在我們的範例中:

<code class="cpp">template <typename ... Types>
class Base {
public:
    template <typename T>
    typename std::enable_if<Contains<T, Types ...>::value>::type
    foo() {
        std::cout << "Base::foo()\n";
    }
};

我們可以衍生一個類別Derived,它從Base 繼承兩次且類型集不重疊:

<code class="cpp">struct Derived: public Base<int, char>,
                public Base<double, void>
{};</code>

呼叫Derived().foo() 時,編譯器無法解析從哪個基底類別呼叫foo(),導致不明確的呼叫錯誤。

為什麼編譯器無法解決歧義

類別的合併規則-成員查找聲明如果派生類別(在本例中為Derived)的聲明集為空,則成員(在本例中為foo)的查找集將從所有直接基類合併。但是,由於我們的基類對 foo 具有不同的聲明集,因此合併是不明確的。

解決方法

要解決歧義,我們可以將Derived 的聲明集設為透過在基類中為foo 方法添加using 聲明來實現非空:

<code class="cpp">struct Derived: public Base<int, char>,
                public Base<double, void>
{
    using Base<int, char>::foo;
    using Base<double, void>::foo;
};</code>

Using 聲明將基類中的成員引入派生類,從而有效地為Derived 提供兩個foo 重載。然後,編譯器可以明確地呼叫適當的重載。

替代解決方案

  • 基礎收集器類別:此類模板可以使用聲明進行收集
  • C 17 中的Pack Expansion: Pack Expansion 可用於將BaseCollector 範本簡化為一行,使其更加簡潔高效編譯。

以上是呼叫多型方法時如何解出型別重疊的多重繼承中的二義性?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn