让我们承认,出于任何正当的理由,动态(虚拟)多态性对我们来说是不可用的。 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的异构集合?
最简单的解决方案(但可能不是最优雅的)是使用std::tuple
。 下面的示例运行良好:
auto coll = std::make_tuple(d1, d2);
std::apply([&](auto & ... el) {
(..., el.sayBaz());
}, coll);
迭代并不是直接的,它具有std::tuple
的所有优点和缺点(主要是没有运行时插入),但至少它能工作!