Mat类型主要是跟matlab中的数据类型一样。故用起来很方便。

Mat最大的优势跟STL很相似,都是对内存进行动态的管理,不需要之前用户手动的管理内存,对于一些大型的开发,有时候投入的lpImage内存管理的时间甚至比关注算法实现的时间还要多,这显然是不合适的。除了有些嵌入式场合必须使用c语言,我任何时候都强烈像大家推荐Mat。

Mat这个类有两部分数据。一个是matrix header,这部分的大小是固定的,包含矩阵的大小,存储的方式,矩阵存储的地址等等。另一个部分是一个指向矩阵包含像素值的指针。

[cpp] view plaincopy
  1. Mat A, C; // creates just the header parts
  2. A = imread(argv[1], CV_LOAD_IMAGE_COLOR); // here we’ll know the method used (allocate matrix)
  3. Mat B(A); // Use the copy constructor
  4. C = A; // Assignment operator

需要注意的是,copy这样的操作只是copy了矩阵的matrix header和那个指针,而不是矩阵的本身,也就意味着两个矩阵的数据指针指向的是同一个地址,需要开发者格外注意。比如上面这段程序,A、B、C指向的是同一块数据,他们的header不同,但对于A的操作同样也影响着B、C的结果。刚刚提高了内存自动释放的问题,那么当我不再使用A的时候就把内存释放了,那时候再操作B和C岂不是很危险。不用担心,OpenCV的大神为我们已经考虑了这个问题,是在最后一个Mat不再使用的时候才会释放内存,咱们就放心用就行了。

如果想建立互不影响的Mat,是真正的复制操作,需要使用函数clone()或者copyTo()。

说到数据的存储,这一直就是一个值得关注的问题,Mat_<uchar>对应的是CV_8U,Mat_<uchar>对应的是CV_8U,Mat_<char>对应的是CV_8S,Mat_<int>对应的是CV_32S,Mat_<float>对应的是CV_32F,Mat_<double>对应的是CV_64F,对应的数据深度如下:

• CV_8U - 8-bit unsigned integers ( 0..255 )

• CV_8S - 8-bit signed integers ( -128..127 )

• CV_16U - 16-bit unsigned integers ( 0..65535 )

• CV_16S - 16-bit signed integers ( -32768..32767 )

• CV_32S - 32-bit signed integers ( -2147483648..2147483647 )

• CV_32F - 32-bit floating-point numbers ( -FLT_MAX..FLT_MAX, INF, NAN )

• CV_64F - 64-bit floating-point numbers ( -DBL_MAX..DBL_MAX, INF, NAN )

这里还需要注意一个问题,很多OpenCV的函数支持的数据深度只有8位和32位的,所以要少使用CV_64F,但是vs的编译器又会把float数据自动变成double型,有些不太爽。

还有个需要注意的问题,就是流操作符<<对于Mat的操作,仅限于Mat是2维的情况。

还有必要说一下Mat的存储是逐行的存储的。

再说说Mat的创建,方式有两种,罗列一下:1.调用create(行,列,类型)2.Mat(行,列,类型(值))。例如:

[cpp] view plaincopy
  1. // make a 7x7 complex matrix filled with 1+3j.
  2. Mat M(7,7,CV_32FC2,Scalar(1,3));
  3. // and now turn M to a 100x60 15-channel 8-bit matrix.
  4. // The old content will be deallocated
  5. M.create(100,60,CV_8UC(15));

要是想创建更高维的矩阵,要写成下面的方式

[cpp] view plaincopy
  1. // create a 100x100x100 8-bit array
  2. int sz[] = {100, 100, 100};
  3. Mat bigCube(3, sz, CV_8U, Scalar::all(0));

对于矩阵的行操作或者列操作,方式如下:(注意对列操作时要新建一个Mat,我想应该跟列地址不连续有关)

[cpp] view plaincopy
  1. // add the 5-th row, multiplied by 3 to the 3rd row
  2. M.row(3) = M.row(3) + M.row(5)*3;
  3. // now copy the 7-th column to the 1-st column
  4. // M.col(1) = M.col(7); // this will not work
  5. Mat M1 = M.col(1);
  6. M.col(7).copyTo(M1);

下面的东西就比较狂暴了,对于外来的数据,比如你从别的地方接受了一幅图片,但可以不是Mat结构的,而只有一个数据的指针,看看接下来的代码是如何应付的,重点哦,亲

[cpp] view plaincopy
  1. void process_video_frame(const unsigned char* pixels,
  2. int width, int height, int step)
  3. {
  4. Mat img(height, width, CV_8UC3, pixels, step);
  5. GaussianBlur(img, img, Size(7,7), 1.5, 1.5);
  6. }

亲,有木有很简单!!!

还有一种快速初始化数据的办法,如下:

[cpp] view plaincopy
  1. double m[3][3] = {{a, b, c}, {d, e, f}, {g, h, i}};
  2. Mat M = Mat(3, 3, CV_64F, m).inv();

也可以把原来的IplImage格式的图片直接用Mat(IplImage)的方式转成Mat结构,也可以像Matlab一样调用zeros()、ones()、eye()这样的函数进行初始化。

如果你需要提前释放数据的指针和内存,可以调用release()。

对于数据的获取,当然还是调用at<float>(3, 3)这样的格式为最佳。其他的方法我甚少尝试,就不敢介绍了。

最后要提的一点是关于Mat的表达式,这个也非常多,加减乘除,转置求逆,我怎么记得我以前介绍过呢。那就不多说啦~

转载于:https://www.cnblogs.com/ljy2013/p/3407551.html

opencv中的Mat类型相关推荐

  1. OpenCV中图像Mat,二维指针和CxImage类之间的转换

    在做图像处理中,常用的函数接口有Opencv中的Mat图像类,有时候需要直接用二维指针开辟内存直接存储图像数据,有时候需要用到CxImage类存储图像.本文主要是总结下这三类存储方式之间的图像数据的转 ...

  2. 如何将OpenCV中的Mat类绑定为OpenGL中的纹理

    https://blog.csdn.net/TTTTzTTTT/article/details/53456324 如果要调用外接的USB摄像头获取图像通常使用OpenCV来调用,如何调用摄像头请参考本 ...

  3. opencv中的Mat图使用CDC显示

    需求:MFC显示opencv读取的Mat图 代码: 1.中间转化的函数: //************************************ // 函数名称: Show2DC // 访问权限 ...

  4. OpenCV中图像Mat存储格式和MATLAB中图像Mat存储格式的区别

    首先,看一下图像中的宽高与笛卡尔坐标系之间的关系如下图所示,即x与width(cols)对应,y与height(rows)对应,x是按列来进行变化,y按行变化. OpenCV读入图像以Mat形式存储时 ...

  5. opencv中查看mat位图的像素幅度(Cv::matStep)

    实例 其中step里的 ,其中数据指针首地址是p=0x000000000028d7b0,1280是每行数据所占的字节数,1是每个元素的字节数. Mat的作用 The class Mat represe ...

  6. C语言使用指针处理opencv中的Mat图像数据

    1.在处理图像时,一般直接使用opencv中的imread函数获取图片,但是获取到图片后没有用到opencv中的其他算法时,直接用图片处理就会出现耗时严重的情况,所以需要将图片形式转换成指针数组形式处 ...

  7. 遍历opencv中的mat像素的几种方法和概念

    今天在看矩形滤波的时候忽然脑子短路,把一些概念全弄混了,现总结一下,以便下次再混的时候可以参考确认下,自己的理解,有错的地方还请指正. 首先,在Opencv2中基本上都是用的Mat来表示图像了,C++ ...

  8. c++版opencv中的Mat数据类型的说明

    一直使用mat,很好用,但是细扣又说不清楚到底是怎样的一种数据类型,今天学习下. 一.先上硬货结论: 浅拷贝:拷贝构造函数和赋值运算符只复制信息头,即实际上还是同个图像数据.mat中存储同个数据地址: ...

  9. 利用FreeImage将gif图像转为opencv中的Mat

    原文:http://www.cnblogs.com/monkeyhey/p/3927857.html 网上有将gif转为iplimg的版本,只是用惯了C++的接口,所以就写了个转Mat的版本,代码比较 ...

最新文章

  1. 蓝桥杯 ADV-144 算法提高 01背包
  2. Java为什么能跨平台运行
  3. [SpringBoot2]原生组件注入_原生注解与Spring方式注入
  4. 排序 -> 插入排序
  5. 列级触发器 SQL Server
  6. node.js(二)创建服务器
  7. 树莓派使用无线网卡上网相关命令
  8. Flask第十八篇 Flask-Migrate
  9. windows安装python3步骤_windows下python3第三方库安装方法总结
  10. X86工控机虚拟显示器设置(nomachine远程桌面)
  11. Java将数字金额转换为中文大写
  12. Android快捷开关实现
  13. 推荐系统的PMF - 概率矩阵分解和协同过滤
  14. 杰奇运行在php7,index.php
  15. 机器学习 决策树和随机森林
  16. 【基于物理的渲染(PBR)白皮书】(五)几何函数相关总结
  17. Java(老白再次入门) - 泛型
  18. 互联网时代,用小众语言对抗焦虑和内卷!
  19. python IEEE OSA GOOGLE学术下载
  20. 【读书笔记】《幸福的方法》——人生的终极财富与意义

热门文章

  1. linux mysql误删,linux下MySQL安装与删除 (Ubuntu)
  2. 力扣面试题 01.08. 零矩阵
  3. 导出配置_Lua配置表导出优化
  4. kotlin 判断数字_Kotlin程序检查数字是偶数还是奇数
  5. Java——集合(输入一串字符串,统计字符串中每个字符出现的次数)
  6. c语言interrupt函数,中断处理函数数组interrupt[]初始化
  7. nginx php fpm 日志,nginx下php-fpm不记录php报错日志怎么办?
  8. python 导入自己写的类_python中自己的类不能被导入
  9. 739. 每日温度 golang
  10. 初始序列为1 8 6 2 5 4 7 3一组数采用堆排序,当建堆(小根堆)完毕时,堆所对应的二叉树中序遍历序列为