std :: копировать трехмерный массив в трехмерный вектор - PullRequest
0 голосов
/ 12 октября 2019

Я пытаюсь скопировать 3D-массив в 3D-вектор, который имеет те же размеры, что и массив. Нет проблем, если я использую вложенный цикл для создания копии, и программа компилирует OK, если я пытаюсь использовать std :: copy, но программа выдает исключение при запуске (нарушение доступа в memcpy.asm в CopyUpDword Loop). Очевидно, я где-то выходил за границы, но почему?

Использование вложенного цикла для копирования массива в вектор и проверка вектора до и после копирования показывает, что вектор имеет правильный размер.

#include <string>
#include <vector>
#include <algorithm>
using namespace std;

typedef vector<string> vsCols;
typedef vector<vsCols> vsRows;
typedef vector<vsRows> vsPage;

string Array[2][3][2];

int main()
{

// dimension the vector same as the array
vsPage pages(2);

for (size_t page = 0; page < pages.size(); page++) {
    pages[page].resize(3);
    for (size_t row = 0; row < pages[page].size(); row++) {
        pages[page][row].resize(2);
    }
}

// fill Array
string s;
    for (int page = 0; page < 2; page++) {
        for (int row = 0; row < 3; row++) {
            for (int col = 0; col < 2; col++) {
        s = "Item" + to_string(((page + 1) + (row + 2)) * (col + 1));
        Array[page][row][col] = s;
            }
        }
    }

// throws the exception described above
copy(&Array[0][0][0], &Array[0][0][0] + 2 * 3 * 2, &pages[0][0][0]);

Синтаксис похож на пример, который я видел здесь, который использовал std :: copy для копирования одного 3D-массива в другой 3D-массив, поэтому я ожидал, что он также будет работать с вектором в качестве места назначения копирования.

1 Ответ

0 голосов
/ 12 октября 2019

A vector из vector s не является смежным. Каждый вектор, и у вас есть 12 из них, не включая vector, соединяющие 12, содержит свой собственный блок динамически выделяемой памяти. &pages[0][0][0] дает адрес первого столбца первой строки первой страницы. Кром знает только, где в памяти находится второй столбец первого ряда первой страницы, но вероятность того, что он будет сразу после первого, удивительно мала. Это почти наверняка нарушение доступа.

С подходом vector из vector s лучшее, на что вы можете надеяться, это что-то вроде:

vsPage pages;
for (const auto & row : Array)
{
    vsRows r;
    for (const auto & col : row)
    {
        r.emplace_back(std::begin(col), std::end(col));
    }
    pages.emplace_back(r);

}

Вы действительно можете только копироватьпо одному столбцу за раз.

Так что, вероятно, пришло время выбросить vector из vector s и сделать vector LOOK похожим на многомерный vector. Это хороший 2-D дизайн , который вы сможете разбить на 3 измерения. Как только все данные будут смежными, вы можете обмануть, как ад и std::copy все данные так, как пытался просящий. Я бы попытался скрыть это за пределами класса в 3-D классе, чтобы вы могли изменить его позже, не мешая никому.

Пример:

// Basic framework gleefully looted from jamesdlin
class ThreeDee
{
public:
    // zero initialized "empty" 3D matrix
    ThreeDee(size_t pages, size_t rows, size_t cols) :
            mPages(pages), mRows(rows), mCols(cols), mData(pages * rows * cols)
    {
    }


    // array initialized "empty" 3D matrix
    // the template params are deduced from the parameters of the array
    // but to get an array rather than a decayed pointer, we need to pass the array 
    // by reference
    template<size_t PAGES, size_t ROWS, size_t COLS>
    ThreeDee(string (&arr)[PAGES][ROWS][COLS]) :
            mPages(PAGES), mRows(ROWS), mCols(COLS), mData(PAGES * ROWS * COLS)
    {
        // I hate this, but I don't have anything better yet and I have to 
        // surrender the computer to a student with homework
        std::copy(&arr[0][0][0], &arr[0][0][0]+mData.size(), mData.data());
    }

    string& operator()(size_t page, size_t row, size_t col)
    {
        return mData[(page * mRows + row) * mCols + col]; // 3D to 1D mapping 
    }

    string operator()(size_t page, size_t row, size_t col) const
    {
        return mData[(page * mRows + row) * mCols + col];
    }

    size_t pages()const
    {
        return mPages;
    }
    size_t rows()const
    {
        return mRows;
    }
    size_t cols()const
    {
        return mCols;
    }

private:
    size_t mPages;
    size_t mRows;
    size_t mCols;
    std::vector<string> mData;
};

Это делает использованиепросто:

string Array[2][3][2];
ThreeDee pages(Array);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...