首页 > 解决方案 > 为什么不能用两层列表初始化器初始化 2D std::array?

问题描述

有人可以帮我理解为什么我的编译器不能/不能推断出这个吗?(使用 g++ 7.3)

不工作:

#include <array>
std::array<std::array<double,2>,2> f() {
 return {{0,0},{0,0}};
}

工作正常:

#include <array>
std::array<std::array<double,2>,2> f() {
 return {std::array<double,2>{0,0},{0,0}};
}

同样奇怪的是,这也失败了:

#include <array>
std::array<std::array<double,2>,2> f() {
 return std::array<std::array<double,2>,2>{{0,0},{0,0}};
}

@1201ProgramAlarm 指出添加另一组花括号有效:

#include <array>
std::array<std::array<double,2>,2> f() {
 return {{{0,0},{0,0}}};
}

它使用聚合初始化,因为std::array没有用于大括号初始化列表的构造函数。很好,但是为什么/如何工作?

std::array<double,2> x{1,2};

为什么它处理这种情况而不是嵌套情况?

标签: c++stdarraylist-initialization

解决方案


容器std::array等价地是一个持有C数组的struct(一个实现可能不会这样实现std::array,但应该保证语义相同),所以应该用两层大括号来初始化,即

#include <array>
std::array<std::array<double,2>,2> f() {
   return {{{{0,0}},{{0,0}}}};
} 

当然,初始化列表中的大括号可以像我们通常对二维数组所做的那样被省略:

int arr[2][2] = {0,1,2,3};

...但是在省略号之前以省略号大括号开头的初始化列表不应以省略号之后的左大括号开头。换句话说,如果初始化列表以左大括号开头,编译器将不会考虑此初始化列表已省略最外层大括号的可能性。

在您的初始化程序{{0,0},{0,0}}中,子初始化程序{0,0},{0,0}以左大括号开头,因此它用于初始化 C 数组本身。但是,列表中有两个子句,而只有一个 C 数组,则会发生错误。

在您的初始化程序{std::array<double,2>{0,0},{0,0}}中,子初始化程序std::array<double,2>{0,0},{0,0}不以左大括号开头,因此可以使用它来初始化 C 数组的元素,这可以(递归地,{0,0}可以初始化 an,std::array<double,2>因为子初始化程序0,0不开始带左大括号)。


一个建议:使用大括号的省略规则,您可以省略所有内部大括号,就像我们通常对二维数组所做的那样:

#include <array>
std::array<std::array<double,2>,2> f() {
   return {0,0,0,0};
} 

推荐阅读