OpenCV(C ++)中的Mat类

时间:2019-03-10 09:21:23

标签: c++ opencv

根据opencv文档,Mat类描述为: Mat基本上是具有两个数据部分的类:矩阵标头(包含诸如矩阵大小之类的信息,用于存储的方法,地址位于(存储的矩阵,依此类推)指向包含像素值的矩阵的指针

有人可以帮助理解什么是标题,以及如何声明类吗?

1 个答案:

答案 0 :(得分:2)

根据OpenCV 2.4.xxx

  

Mat基本上是a class with two data partsthe matrix header(包含诸如矩阵大小,用于存储的方法,在哪个地址存储矩阵等信息)和a pointer to the matrix containing the pixel values(根据选择的存储方法采用任意尺寸)。 matrix header size is constant,但是矩阵本身的大小可能因图像而异,通常会大几个数量级。

一个简单的公式:a Mat object = the matrix header + the matrix data pointer

好,什么是矩阵数据指针?是uchar* data,它指向矩阵数据。

然后将cv::Mat中的所有其他人称为matrix header

两个部分的优点是什么?我们可以对矩阵进行浅表复制,并使用引用计数器进行内存管理。至于reference counter(counting),这是编程中的重要主题。来自Wiki reference counter(counting)In computer science, reference counting is a technique of storing the number of references, pointers, or handles to a resource such as an object, block of memory, disk space or other resource.


cv::Mat中有两个重要的功能供您参考:

void cv::Mat::addref() and void cv::Mat::release()

/** @brief Increments the reference counter.
The method increments the reference counter associated with the matrix data. If the matrix header
points to an external data set (see Mat::Mat ), the reference counter is NULL, and the method has no
effect in this case. Normally, to avoid memory leaks, the method should not be called explicitly. It
is called implicitly by the matrix assignment operator. The reference counter increment is an atomic
operation on the platforms that support it. Thus, it is safe to operate on the same matrices
asynchronously in different threads.
 */
void addref();

/** @brief Decrements the reference counter and deallocates the matrix if needed.
The method decrements the reference counter associated with the matrix data. When the reference
counter reaches 0, the matrix data is deallocated and the data and the reference counter pointers
are set to NULL's. If the matrix header points to an external data set (see Mat::Mat ), the
reference counter is NULL, and the method has no effect in this case.
This method can be called manually to force the matrix data deallocation. But since this method is
automatically called in the destructor, or by any other method that changes the data pointer, it is
usually not needed. The reference counter decrement and check for 0 is an atomic operation on the
platforms that support it. Thus, it is safe to operate on the same matrices asynchronously in
different threads.
 */
void release();

当然,也许您只需要注意:

cv::Mat a = cv::Mat::zeros(2,2,CV_8UC1);
cv::Mat b,c;

b = a;       // shallow copy, share the same matrix data by the data pointer
a.copyTo(c); // deep copy, allocate matrix data memory, and assign the new pointer 

演示:

#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;

int main(){
    cv::Mat a = cv::Mat::zeros(2,2,CV_8UC1);
    cv::Mat b,c;

    b = a;       // shallow copy, share the same matrix data pointer
    a.copyTo(c); // deep copy, allocate matrix data memory, and assign the new pointer

    std::cout << "----- a -----\n" << a << "\n----- b -----\n" << b << "\n----- c -----\n" << c << std::endl;


    std::cout << "\nModify a, b, c:\n";
    a.at<unsigned char>(0,0) = 1; // a, b share the same matrix data
    b.at<unsigned char>(1,0) = 2;
    c.at<unsigned char>(1,1) = 3; // c has independent matrix data

    std::cout << "----- a -----\n" << a << "\n----- b -----\n" << b << "\n----- c -----\n" << c << std::endl;


    return 0;

}

结果:

----- a -----
[  0,   0;
   0,   0]
----- b -----
[  0,   0;
   0,   0]
----- c -----
[  0,   0;
   0,   0]

Modify a, b, c:
----- a -----
[  1,   0;
   2,   0]
----- b -----
[  1,   0;
   2,   0]
----- c -----
[  0,   0;
   0,   3]