所以我遇到了一些让我非常困扰的事情,让我试着弄清楚为什么它会这样工作:

如果有以下 super 简单、容易出错的代码只是为了展示一个例子:

std::vector<cv::Mat> newData(3,cv::Mat(height, width, cv::DataType<T>::type));

int counter = 0;
for(int b=0; b<3; b++){
    for(int i=0; i<3; i++){
        for(int j=0; j<3; j++){
            newData[b].at<int>(i,j) = counter++;
            std::cout << newData[b].at<T>(i,j) << std::endl;
        }
    }
}

for(int b=0; b<3; b++){
    std::cout << newData[b] << std::endl;
}

打印:
[18, 19, 20;
  21, 22, 23;
  24, 25, 26]
[18, 19, 20;
  21, 22, 23;
  24, 25, 26]
[18, 19, 20;
  21, 22, 23;
  24, 25, 26]

为什么对不同的 vector 条目使用相同的引用?
我被迫分别创建不同的矩阵,而不是与 vector 本身一起创建它们。

有没有办法避免这种情况?

谢谢

最佳答案

问题是 cv::Mat 具有引用语义,因此复制 cv::Mat 对象会导致拷贝与原始共享数据。因此以这种方式初始化 vector

std::vector<cv::Mat> newData(N, a_cv_mat);

将导致包含 N cv::Mat 的 vector 都与 a_cv_mat 共享相同的数据。

为了避免 cv::Mat 对象引用相同的数据,您可以使用大括号括起来的初始化列表来初始化 vector :
std::vector<cv::Mat> newData{cv::Mat(height, width, cv::DataType<T>::type),
                             cv::Mat(height, width, cv::DataType<T>::type),
                             cv::Mat(height, width, cv::DataType<T>::type)};

如果你在编译时不知道元素的数量,你可以将它们一一放入 vector 中:
std::vector<cv::Mat> newData
newData.emplace_back(height, width, cv::DataType<T>::type);

如果您没有 C++11 支持,您可以将每个矩阵插入 vector 中:
std::vector<cv::Mat> newData
newData.push_back(cv::Mat(height, width, cv::DataType<T>::type));

关于c++ - std::vector 不会为多个 vector 条目创建新的 cv::Mat 引用 - 初始化矩阵时数据被覆盖,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/26315928/

10-16 22:51