提问者:小点点

基于CRTP的多态性元件的异质集合


让我们承认,出于任何正当的理由,动态(虚拟)多态性对我们来说是不可用的。 CRTP提供了如下所示的高效静态多态性:

template<typename derived_t>
struct Base {
    void sayFoo() { std::cout << "foo" << std::endl; }
    void sayBaz() { std::cout << impl().bazWord() << std::endl; } 
        
    derived_t& impl() { return static_cast<derived_t&>(*this); }
   
};

struct derived1 : public Base<derived1> {
    std::string bazWord() { return "baz1"; }
};

struct derived2 : public Base<derived2> {
    std::string bazWord() { return "baz2"; }
};

实例化很容易完成,如下所示:

derived1 d1;
d1.sayFoo(); // says "foo"
d1.sayBaz(); // says "baz1"

derived2 d2;
d1.sayFoo(); // says "foo"
d2.sayBaz(); // says "baz2"

现在,我想将这些元素保存在一个异质集合中,如下所示:

std::vector<Base<??>> v;
v.push_back(d1);
v.push_back(d2);
std::for_each(v.begin(), v.end(), [](auto& elem) { elem.sayBaz(); })

这样做会给我一个错误,显然是因为无法定义基类型。

有没有一种简单的方法来实现基于CRTP的异构集合?


共1个答案

匿名用户

最简单的解决方案(但可能不是最优雅的)是使用std::tuple。 下面的示例运行良好:

auto coll = std::make_tuple(d1, d2);
std::apply([&](auto & ... el) { 
    (..., el.sayBaz());
}, coll);

迭代并不是直接的,它具有std::tuple的所有优点和缺点(主要是没有运行时插入),但至少它能工作!