如何从嵌套的std :: initializer_list初始化2D C样式数组?

时间:2019-09-18 18:45:06

标签: c++ c++11 templates multidimensional-array stdinitializerlist

我正在创建一个Matrix<type, width, height>类,我希望能够使用initializer_list进行初始化,例如:

Matrix<int, 2, 2> mat = Matrix<int, 2, 2>{ {1,2},{3,4} };

使用T[height][width] 2D数组实现矩阵。

为此,我尝试制作一个构造器,例如:

Matrix(std::initializer_list< std::initializer_list<T>> input);

但是,我不知道如何用列表填充数组。我尝试过

  • 使用memcpy(损坏的内存),

  • std::copy(似乎无法从 std::initializer_list

  • 以及赋值运算符()构造函数调用 (但2D数组​​不可分配,并且()构造函数调用不会 编译)。

还有其他方法,最好是尽可能安全吗?

1 个答案:

答案 0 :(得分:3)

您应为Matrix(std::initializer_list</* here */> list);提供准确的类型。 然后,要填充数组,您需要遍历传递的列表并填充数组。 See live online

#include <initializer_list> // std::initializer_list

template<typename T, std::size_t R, std::size_t C>
class Matrix
{
    T mArray2D[R][C]{};

public:
    Matrix(const std::initializer_list<T[C]> list)
    //                                 ^^^^^ --> type of the row
    {
        auto iter{ list.begin() }; // returns pointer to T[C]
        for (std::size_t row{}; row < R; ++row)
        {
            const auto& rowElements{ *iter };
            for (std::size_t col{}; col < C; ++col)
            {
                mArray2D[row][col] = rowElements[col];
            }
            ++iter;
        }
    }
};

现在,在main()中,您可以使用初始化器列表来初始化Matrix,如下所示:

Matrix<int, 2, 2> mat{ {1,2}, {3,4} };
// or
// Matrix<int, 2, 2> mat = { {1,2}, {3,4} };

如果可以使用std::array,则可以执行以下操作。 See live online

#include <iostream>
#include <array>            // std::array
#include <initializer_list> // std::initializer_list

template<typename T, std::size_t R, std::size_t C>
class Matrix
{
    using RowType = std::array<T, C>;
    T mArray2D[R][C]{};

public:
    Matrix(const std::initializer_list<RowType> list)
    {
        auto iter{ list.begin() };
        for (std::size_t row{}; row < R; ++row)
        {
            const RowType& rowElements{ *iter };
            for (std::size_t col{}; col < C; ++col)
            {
                mArray2D[row][col] = rowElements[col];
            }
            ++iter;
        }
    }
};

如果类中的多维数组可以是类型为std::arraystd::array T,则也可以使用std::copySee live online

#include <iostream>
#include <array>            // std::array
#include <initializer_list> // std::initializer_list
#include <algorithm>        // std::copy

template<typename T, std::size_t R, std::size_t C>
class Matrix
{
    using RowType = std::array<T, C>;
    using Array2D = std::array<RowType, R>;
    Array2D mArray{ RowType{} }; // initialize the matrix

public:
    Matrix(const std::initializer_list<RowType> list)
    {
       std::copy(list.begin(), list.end(), mArray.begin());
    }
};