使用作为参数传递的 std::intializer_list 初始化 std::array

Initializing an std::array with an std::intializer_list passed as a parameter

我正在用 C++ 实现一个 Matrix class,我希望能够像这样用一个初始化器列表来初始化它:

Matrix<double, 2, 3> m({{1,2,1}, 
                        {0,1,1}});

在 class 中,我实现了这样的构造函数:

template<typename Ty, unsigned int rows, unsigned int cols>
class Matrix
{
    std::array<std::array<Ty, cols>, rows> data;
public:
    Matrix(const std::initializer_list<std::initializer_list<Ty>>& list)
    {
        int i = 0;
        for (auto& list_row : list)
        {
            int j = 0;
            for (auto& list_value : list_row)
            {
                data[i][j++] = list_value;
            }
            i++;
        }
    }
}

代码有效,但我有几个问题。是否有可能使其安全,因此它只需要一定大小(矩阵的大小)的初始化列表,其次我可以使代码更简洁 - 我曾想象过这样的事情:

Matrix(const std::initializer_list<std::initializer_list<Ty>>& list)
    : data(list)
{ }

它不起作用,但如果有更好的方法,我们将不胜感激。

有一种非常简单的方法可以确保安全:

template<typename Ty, unsigned int rows, unsigned int cols>
class Matrix
{
    std::array<std::array<Ty, cols>, rows> data;
public:
    Matrix(std::array<std::array<Ty, cols>, rows> const& in)
        : data(in)
    { }
};

这让您几乎 使用您想要的语法 - 您只需要一组额外的 {}s。

无法静态地强制执行 std::initializer_list 的大小 - 您必须 assertthrow 以防止我传递一堆额外的值并导致你执行超出范围的写入。