模板结构体的大括号初始化列表

4
#include <array>                                                                
#include <vector>                                                               
#include <cinttypes>                                                            
#include <iostream>                                                             

using namespace std;                                                            

template<size_t N>                                                              
struct item_t {                                                                 
  array<uint32_t, N> weight = {0};                                              
};                                                                              

int main(void) {                                                                

  vector<item_t<3>> items;                                                      
  items.emplace_back({{9,2,3}});                                                
  cout << items[0].weight[0] << endl;                                           
  return 0;                                                                     
};  

我有点不知所措了。错误出现在emplace_back这一行,不知道该如何解决。任何帮助或提示都将不胜感激,谢谢。

编辑

gcc版本为4.8.2

$ g++ -std=c++11 test.cpp 
test.cpp: In function ‘int main()’:
test.cpp:16:30: error: no matching function for call to ‘std::vector<item_t<3ul> >::emplace_back(<brace-enclosed initializer list>)’
  items.emplace_back({{9,2,3}});
                              ^
test.cpp:16:30: note: candidate is:
In file included from /usr/include/c++/4.8/vector:69:0,
                 from test.cpp:2:
/usr/include/c++/4.8/bits/vector.tcc:91:7: note: void std::vector<_Tp, _Alloc>::emplace_back(_Args&& ...) [with _Args = {}; _Tp = item_t<3ul>; _Alloc = std::allocator<item_t<3ul> >]
       vector<_Tp, _Alloc>::
       ^
/usr/include/c++/4.8/bits/vector.tcc:91:7: note:   candidate expects 0 arguments, 1 provided

问题出在结构体初始化的= {0}emplace_back上。

你得到了什么错误? - deW1
可以通过删除向量并仅将花括号初始化器分配给item_t<3>来简化它。 - Gareth A. Lloyd
2个回答

7

emplace_back()使用模板参数推断来确定传递给函数的元素类型。 大括号包含的初始化列表不是表达式,没有类型,因此无法通过模板推断。这里必须显式调用构造函数:

items.emplace_back(item_t<3>{{1,2,3}});

这有助于 emplace_back,如果去掉 = {0} 就可以工作。谢谢。 - Gareth A. Lloyd

3
这里有两个问题:
尝试像这样初始化类型为T的对象T{...}被称为聚合初始化。在某些情况下,即使您没有接受initializer_list的构造函数,也会为其指定默认行为。在C++11中,您不允许提供非默认构造函数或类内初始值设定项。因此,考虑到这个定义:
template<size_t N>                                                              
struct item_t {                                                                 
  array<uint32_t, N> weight   = {0};                                            
};

您不能写item_t<3> t{1,2,3};

然而,这不是您的问题。您的代码失败的原因是emplace_back尝试将参数转发到vector底层类型的构造函数中。在您的情况下,没有匹配项。请注意,在此上下文中,漂亮的花括号初始化列表与initializer_list不等价,您无法通过添加initializer_list构造函数来解决此问题,您需要以其他方式帮助编译器。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接