26
struct SS {int a; int s;};

int main ()
{
   vector<SS> v;
   v.push_back(SS{1, 2});
}

代码可以编译没有任何错误。但是,当在类中初始化结构时,出现编译错误。谁能解释一下?

struct SS {int a = 0; int s = 2;};

错误:

In function ‘int main()’:
error: no matching function for call to ‘SS::SS(<brace-enclosed initializer list>)’
     v.push_back(SS{1, 2});
                        ^
note: candidates are:
note: constexpr SS::SS()
 struct SS {int a = 0; int s = 2;};
        ^
note:   candidate expects 0 arguments, 2 provided
note: constexpr SS::SS(const SS&)
note:   candidate expects 1 argument, 2 provided
note: constexpr SS::SS(SS&&)
note:   candidate expects 1 argument, 2 provided
4

3 回答 3

38

在 C++11 中,当您在声明点使用非静态数据成员初始化时,如下所示:

struct SS {int a = 0; int s = 2;};

你让这个类成为一个非聚合的。这意味着您不能再像这样初始化实例:

SS s{1,2};

要使此初始化语法适用于非聚合,您必须添加一个双参数构造函数:

struct SS 
{
  SS(int a, int s) : a(a), s(s) {}
  int a = 0; 
  int s = 2;
};

此限制已在 C++14 中解除。

请注意,您可能希望为该类添加一个默认构造函数。用户提供的构造函数的存在禁止编译器生成默认构造函数。

请参阅此处的相关阅读。

于 2013-08-12T10:00:40.480 回答
5

使用默认成员初始值设定项将类/结构呈现为非聚合:

§ 8.5.1 聚合

聚合是一个数组或一个类(第 9 条),没有用户提供的构造函数(12.1),没有用于非静态数据成员的大括号或等号初始化器(9.2),没有私有或受保护的非静态数据成员(第 11 条),没有基类(第 10 条),也没有虚函数(10.3)。

聚合和非聚合的语义不同:

聚合(例如,数组和结构):

Initialize members/elements beginning-to-end.

非聚合:

Invoke a constructor.

v.push_back(SS{1, 2}); // Error, it tries to call SS constructor

这意味着您现在需要一个构造函数:

struct SS 
{
  SS(int a, int s) : a(a), s(s) 
  {
  }
  int a = 0; 
  int s = 2;
};
于 2013-08-12T10:05:52.063 回答
0

我有同样的问题。在我的例子中,我有两个structs,它们都有一些构造函数,包括复制构造函数,它们继承自抽象父级。

当上面的建议没有帮助时,我终于意识到我需要explicit从复制构造函数中删除说明符并删除了错误。

我想我会分享,以防另一个可怜的灵魂像我刚才那样花很长时间发现这个错误。

于 2020-05-01T02:13:55.957 回答