C++为类模板提供初始化列表构造函数

cfa*_*771 5 c++ initializer-list make-shared variadic-templates c++11

我有一个带有模板参数T的类模板Templ,而Templ类有一个类型为T的数据成员,称为obj.我编写了一个可变参数构造函数模板,它将参数转发给obj的构造函数:

template <class T>
class Templ
{
public:
     template <class... Args> explicit Templ (Args&&... args)
     : obj (std::forward<Args>(args)...)
     {
     }
private:
     T obj;
};
Run Code Online (Sandbox Code Playgroud)

现在我意识到类型T可能是一个带有init-list构造函数的类,我希望它可以通过Templ访问.所以我检查了什么std::list::emplace,std::make_shared做了什么.它们具有像我一样的可变函数,但它们没有覆盖init-list的覆盖.由于某些原因.

所以第一个问题:为什么?我的意思是,如果我使用一个带有init-list ctor的类T,然后我用std::list<T>?为什么list ::\templates没有带初始化列表的版本?也许有一个很好的理由我应该这样做...所以我想知道.

此外,无论STL做什么 - 我应该提供一个init-list ctor作为好的设计吗?我的意思是,它就像可变的ctor,对吗?允许用户选择任何类型或种类的T,带TEMPL <>使用,并且直接调用为T.定义的任何构造函数即使它是一个构造函数采取一个init-列表.

eca*_*mur 4

转发构造函数的问题initializer_list是,除了最简单的参数类型之外的所有参数类型都是不可推导的(模板并不总是猜测初始值设定项列表类型):

#include <map>
template<typename T> struct U {
   T t;
   template<typename...A> explicit U(A&&...a): t(std::forward<A>(a)...) {}
   template<typename L, typename = typename std::enable_if<
      std::is_constructible<T, std::initializer_list<L>>::value>::type>
      explicit U(std::initializer_list<L> l): t(l) {}
};
U<std::map<int, int>> m{{{0, 1}, {2, 3}}};  // fails, couldn't deduce 'L'
Run Code Online (Sandbox Code Playgroud)

由于在大多数情况下您必须编写m{std::initializer_list<...>{...}},因此仅为基元提供它没有多大意义,而且标准当然也不会这样做。

如果您认为任何有趣的initializer_list参数可能适用于容器类型,您可以查看可选地支持模板的初始值设定项列表构造可能包装容器中采用的方法。