STL push_back:在推送动态变化的数组时修改以前的值

STL push_back: previous values getting modified when pushing a dynamically changing array

本文关键字:数组 修改 变化 动态 back push STL      更新时间:2023-10-16

我的CPP代码中的std::vector.push_back((出现了一个非常奇怪的问题。基本上,我所做的就是:

void func()
{
  if(first_time_coming_here)
  {
    do_some_stuff();
    V.push_back(Mat::zeros(3,1,CV_32FC1));  // OpenCV Mat structure
    V.push_back(Mat_array_that_gets_updated_in_another_function);
  }
  else
  {
    do_other_kinds_of_stuff();
    V.push_back(Mat_array_that_gets_updated_in_another_function);   
  }
}

假设在上一个函数中更新的数组最初是[1,1,1],然后第二次执行func((时,它已经变成了[2,2,2]。我的输出应该是

V = [ [0,0,0], [1,1,1], [2,2,2] ]

但取而代之的是

V = [ [0,0,0], [2,2,2], [2,2,2] ]

我真的对这里发生的事情感到困惑。唯一有意义的是,如果V存储的是数组的CURRENT值,而不是已经传递的值:但一旦调用push_back((,无论引入值的变量发生了什么,元素都不应该获取该值并将其保存在内存中吗?V被定义为cv::Mat的向量。

EDIT:这已经通过清除两个函数调用之间的数组值来解决,如:
array=Mat::零(1,1,CV_32FC1(
然而,我想知道为什么直接使用它不起作用。

您做了一件危险的事情,您推送了一个对象的副本,而不考虑该对象内部的指针或指针指向的内存。您推送的对象已经从原始对象中按成员方式分配了数据成员,因此一旦原始对象消失,您的程序就会崩溃。

解决方案是推送对象的克隆:

V.push_back(Mat_array_that_gets_updated_in_another_function.clone ());

来自Mat:的OpenCV文档

在右侧可以有数组或表达式的位置使用复制构造函数或赋值运算符(请参见下文(。如引言中所述,数组赋值是O(1(运算,因为它只复制标头并增加引用计数器。Mat::clone((方法可用于在需要时获取数组的完整(深层(副本。

您也可以在文档中看到Mat有一个成员uchar* data其中存储矩阵的实际数据条目。

由于您没有对数组进行深度复制,对原始数组的更改也会影响您之前放入vector中的"副本",只要它们没有改变data成员指向的位置。(显然他们没有。(但当你做类似array = Mat::zeros(1,1,CV_32FC1);的事情时原来的Mat似乎得到了一个新的data指针(或者至少从你说不改变已经在vector中的Mat(。