簡體   English   中英

通過類型獲取具有遞歸繼承的可變參數模板的陰影成員

[英]Get shadowed members of variadic template with recursive inheritance by type

我有類似以下內容:

template <typename T>
class A {
public:
    typedef T Type;
};

template <typename...> class B;

template <typename TFirst, typename... TRest>
class B<TFirst, TRest...> : public B<TRest...> {
public:
    typedef typename TFirst::Type Type;

    TFirst a;

    virtual void func(TFirst a, Type t) = 0;
};

template <>
class B<> {};

struct X {};
struct Y {};

class ASub1 : public A<X> {};
class ASub2 : public A<Y> {};

class BSub : public B<ASub1, ASub2> {
public:
    // Implement all pure virtual methods
    // void func(ASub1 a1, Type t) override {}
    // void func(ASub2 a2, Type t) override {}
};

現在我有兩個問題:

  1. 我將如何在BSub實現所有純虛擬方法? 我以某種方式需要訪問所有ASubX的模板參數(類型)。

  2. 是否可以通過傳遞ASubX來訪問所有成員a 我的意思是這樣get<ASub1>(b_sub)如果b_sub是實例BSub

我希望使用C ++ 14解決方案。

對於(1),您可以編寫

class BSub : public B<ASub1, ASub2> {
public:
    void func (ASub1, typename ASub1::Type) override {}
    void func (ASub2, typename ASub2::Type) override {}
};

或者,記住ASub1::TypeXASub2::TypeY

class BSub : public B<ASub1, ASub2> {
public:
    void func (ASub1, X) override {}
    void func (ASub2, Y) override {}
};

對於(2),根據WF的建議,您可以使用

b_sub.B<ASub1, ASub2>::a

訪問ASub1組件,並

b_sub.B<ASub2>::a

訪問ASub2一個。

以下是完整的工作(簡化)示例

#include <iostream>
#include <type_traits>

template <typename T>
struct A
 { using Type = T; };

template <typename...>
struct B;

template <typename TFirst, typename ... TRest>
struct B<TFirst, TRest...> : public B<TRest...>
 {
    using Type = typename TFirst::Type;

    TFirst a;

    virtual void func (TFirst a, Type t) = 0;
 };

template <>
struct B<> {};

struct X {};
struct Y {};

struct ASub1 : public A<X> {};
struct ASub2 : public A<Y> {};

struct BSub : public B<ASub1, ASub2>
 {
   void func (ASub1, X) override { std::cout << 1 << std::endl; }
   void func (ASub2, Y) override { std::cout << 2 << std::endl; }
 };

int main()
 {
   BSub bs;

   bs.func(ASub1{}, X{});
   bs.func(ASub2{}, Y{});

   using T1 = decltype(bs.B<ASub1, ASub2>::a);
   using T2 = decltype(bs.B<ASub2>::a);

   static_assert(std::is_same<T1, ASub1>{}, "!");
   static_assert(std::is_same<T2, ASub2>{}, "!");
 }

std::get for std::tuple實現的啟發,我想出了以下解決方案。

謝謝你的幫助。

#include <cstdio>

template <typename T>
class A {
public:
    using Type = T;
};

template <typename...> class B;

template <typename TFirst, typename... TRest>
class B<TFirst, TRest...> : public B<TRest...> {
public:
    using AType = typename TFirst::Type;
    using Type = B<TFirst, TRest...>;

    TFirst a;

    virtual void func(TFirst a, AType t) {}
};

template <>
class B<> {};

struct X {};
struct Y {};

class ASub1 : public A<X> { public: int value = 10; };
class ASub2 : public A<Y> { public: int value = 20; };

class BSub : public B<ASub1, ASub2> {
public:
    // Implement all pure virtual methods
    void func(ASub1 a1, typename ASub1::Type t) {}
    void func(ASub2 a2, typename ASub2::Type t) {}
};

template <typename TA, typename TB> class BElement;

template <typename TA, typename TAFirst, typename... TARest>
class BElement<TA, B<TAFirst, TARest...>> : public BElement<TA, B<TARest...>> {
public:
};

template <typename TA, typename... TARest>
class BElement<TA, B<TA, TARest...>> {
public:
    using AType = TA;
    using BType = B<TA, TARest...>;
};

template <typename TA, typename... TAs>
TA& get(B<TAs...> & b) {
    using BType = typename BElement<TA, B<TAs...>>::BType;
    return static_cast<BType &>(b).a;
}

template <typename TA, typename TB>
TA& get(TB b) {
    return get<TA>(static_cast<typename TB::Type &>(b));
}

int main() {
  BSub bsub;
  printf("%d\n", get<ASub1>(bsub).value);
  printf("%d\n", get<ASub2>(bsub).value);

  return 0;
}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM