提问者:小点点

两个mats的OpenCV数据相同,但使用MAT::AT检索时的值已损坏


我使用自定义函数序列化然后反序列化一个Mat。 当您比较两个函数的uchar*mat.data时,它们是相同的,但是当我尝试在第三个元素之后直接使用mat.at检查值时,元素值会被卡在一些损坏的浮点值上。 我尝试将元素更改为其他值,但没有任何改变。 当我序列化-反序列化图像垫和直方图垫时也不会发生这个问题。 下面是代码,下面是输出。

cv::Mat kernelx = (cv::Mat_<float>(3, 3) << 1, 0, -1, 1, 0, -1, 1, 0, -1);
std::string stroper = dbop::serializeMat(kernelx);
cv::Mat matoper = dbop::deserializeMat(stroper);
for (int i = 0; i < matoper.rows; i++) {
    for (int j = 0; j < matoper.cols; j++) {
        float f1 = (float)kernelx.data[i + j];
        float f2 = (float)matoper.data[i + j];
        std::cout << kernelx.at<float>(i, j) << " " << matoper.at<float>(i, j) << " " << f1 << " " << f2 << " " << std::endl;
    }
}

下面是输出。 第一个值是原始Mat的元素,第二个值是序列化-反序列化Mat的元素,第三个值是原始Mat被强制转换为浮点的数据值,第四个值是s-d Mat的数据值。

1       1               0       0
0       0               0       0
-1      -4.31596e+08    128     128
1       -4.31602e+08    0       0
0       -4.31602e+08    128     128
-1      -4.31602e+08    63      63
1       -4.31602e+08    128     128
0       -4.31602e+08    63      63
-1      -4.31602e+08    0       0

我猜serializeMat和deserializeMat函数在这里是相关的,所以这里都是。

std::string dbop::serializeMat(cv::Mat operand) {
    std::ostringstream srlzstrstream;
    uchar* pixelPtr = (uchar*)operand.data;
    int cn = operand.channels();
    cv::Scalar_<uchar> cnPixel;

    srlzstrstream << operand.dims << " ";
    for (int i = 0; i < operand.dims; i++) {
        srlzstrstream << operand.size[i] << " ";
    }
    srlzstrstream << operand.type() << " ";

    for (int i = 0; i < operand.total(); i++) {
            for (int k = 0; k < cn; k++) {
                srlzstrstream << (float)pixelPtr[i * cn + k] << " ";
            }
    }
    std::string srlzdstr = srlzstrstream.str();
    return srlzdstr;
}
cv::Mat dbop::deserializeMat(std::string operand){
    std::istringstream desrlzstrstream(operand);
    int mdims, mtype;

    desrlzstrstream >> mdims;
    int* msize = new int[mdims];
    for (int i = 0; i < mdims; i++)
        desrlzstrstream >> msize[i];
    desrlzstrstream >> mtype;

    cv::Mat matoper(mdims, msize, mtype);
    delete [] msize;

    uchar* pixelPtr = (uchar*)matoper.data;
    float cnpixoper;

    for (int i = 0; i < matoper.total(); i++) {
        for (int k = 0; k < matoper.channels(); k++) {
            desrlzstrstream >> cnpixoper;
            pixelPtr[i * matoper.channels() + k] = cnpixoper;
        }
    }
    return matoper;
}

共1个答案

匿名用户

SerializeMat中,

uchar* pixelPtr = (uchar*)operand.data;
...
srlzstrstream << (float)pixelPtr[i * cn + k] << " ";

1,0,-1,1,0,-1,1,0,-1操作数中存储为float,因此应将操作数.data强制转换为float*

还应该将DeserializeMat中的matoper.data强制转换为float*

1 1 0 0 
0 0 0 0 
-1 -1 128 128 
1 1 0 0 
0 0 128 128 
-1 -1 63 63 
1 1 128 128 
0 0 63 63 
-1 -1 0 0