OPENCV_Mat类存取方法元素访问_第1页
OPENCV_Mat类存取方法元素访问_第2页
OPENCV_Mat类存取方法元素访问_第3页
OPENCV_Mat类存取方法元素访问_第4页
OPENCV_Mat类存取方法元素访问_第5页
已阅读5页,还剩3页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、-. z.Opencv-Mat类cv:Mat depth/dims/channels/step/data/elemSize Mat矩阵中数据元素的地址计算公式: addr(Mi0,i1,im-1) = M.data + M.step0 * i0 + M.step1 * i1 + + M.stepm-1 * im-1。其中 m = M.dims 是指M的维度data:Mat对象中的一个指针,指向存中存放矩阵数据的一块存 (uchar* data).row: 行;col:列;rows:行数;cols:列数。dims :Mat所代表的矩阵的维度,如 3 * 4 的矩阵为 2 维,3 * 4 * 5

2、的为3维.channels:通道,矩阵中的每一个矩阵元素拥有的值的个数,比方说 3 * 4 矩阵中一共 12 个元素,如果每个元素有三个值,则就说这个矩阵是 3 通道的,即 channels = 3。常见的是一彩色图片有红、绿、蓝三个通道。但是opencv用imreadopencv读图的函数读进来的图像,三通道存放顺序为B、G、R。depth:深度,即每一个像素的位数(bits),在opencv的Mat.depth()中得到的是一个 0 6 的数字,分别代表不同的位数:enum CV_8U=0, CV_8S=1, CV_16U=2, CV_16S=3, CV_32S=4, CV_32F=5,

3、CV_64F=6 ; 可见 0和1都代表8位, 2和3都代表16位,4和5代表32位,6代表64位; step:是一个数组,定义了矩阵的布局,具体见下面图片分析,另外注意 step1 (step / elemSize1),M.stepm-1 总是等于 elemSize,M.step1(i)返回的是第i维的步长,因此M.step1(m-1)总是等于 channels,m是M的维度;这里是解释步长stepk的,步长也可以看作是与第k维的存储单位,在2维的矩阵中,因为存储是按照行的顺序存储的,整个矩阵存储为一个平面,所以第k=0维的步长也就是单位肯定就是一行所占的字节数;如果是3维的话,第0维是按照

4、面为单位来存储的,第1维是按照行为单位来存储的,第2维是按照元素类型为单位存储的,每个元素类型是根本类型(即uchar,float,short等等)与通道数的乘积.;也就是根本数据类型与通道数组成元素,多个元素组成了行,多行组成了面,多个面组成了3维体,多个3维体 组成4维超体。以此类推,如此看来*一维的步长应该等于高一维的步长step*低一维的大小size。elemSize : 矩阵中每一个元素的数据大小,如果是n通道,就是n*数据类型。如果Mat中的数据的数据类型是 CV_8U 则 elemSize = 1,CV_8UC3 则 elemSize = 3,CV_16UC2 则 elemSiz

5、e = 4;记住另外有个 elemSize1 表示的是矩阵中数据类型的大小,即 elemSize / channels 的大小。图片分析1:考虑二维情况stored row by row按行存储上面是一个 3 * 4 的矩阵,假设其数据类型为 CV_8U,也就是单通道的 uchar 类型这是一个二维矩阵,则维度为 2 M.dims = 2; M.rows = 3; M.cols = 4; sizeof(uchar) = 1,则每一个数据元素大小为 1 M.elemSize() = 1, M.elemSize1() = 1; CV_8U 得到 M.depth() = 0, M.channels(

6、) = 1; 因为是二维矩阵,则 step 数组只有两个值, step0 和 step1 分别代表一行的数据大小和一个元素的数据大小,则 M.step0 = 4, M.step1 = 1; M.step1(0) = M.cols = 4; M.step1(1) = 1; 假设上面的矩阵数据类型是 CV_8UC3,也就是三通道M.dims = 2; M.channels() = 3;M.depth() = 0; M.elemSize() = 3 每一个元素包含3个uchar值 M.elemSize1() = 1 elemSize / channels M.step0 = M.cols * M.e

7、lemSize() = 12, M.step1 = M.channels() * M.elemSize1() = M.elemSize() = 3; M.step(0) = M.cols * M.channels() = 12 ; M.step(1) = M.channels() = 3; 图片分析2:考虑三维情况stored plane by plane按面存储上面是一个 3 * 4 * 6 的矩阵,假设其数据类型为 CV_16SC4,也就是 short 类型M.dims = 3 ; M.channels() = 4 ; M.elemSize1() = sizeof(short) = 2 ;

8、 M.rows = M.cols = 1; M.elemSize() = M.elemSize1() * M.channels() = M.stepM.dims-1 = M.step2 = 2 * 4 = 8; M.step0 = 4 * 6 * M.elemSize() = 192; M.step1 = 6 * M.elemSize() = 48; M.step2 = M.elemSize() = 8; M.step1(0) = M.step0 / M.elemSize() = 48 / 2 = 96 第一维度(即面的元素个数) * 通道数; M.step1(1) = M.step1 / M

9、.elemSize() = 12 / 2 = 24第二维度(即行的元素个数/列宽) * 通道数; M.step1(2) = M.step2 / M.elemSize() = M.channels() = 4第三维度(即元素) * 通道数; End 以上为Mat的存放形式以下为Mat的一些操作方法具体使用方法Fn 1 :利用step。Code 1:int main()/新建一个uchar类型的单通道矩阵grayscale image 灰度图 Mat m(400, 400, CV_8U, Scalar(0);for (int col = 0; col 400; col+) for (int row

10、 = 195; row 205; row+) /获取第row,col个像素点的地址并用 * 符号解析 *(m.data + m.step0 * row + m.step1 * col) = 255; imshow(canvas, m); cvWaitKey();return 0;Output 1 :Code1只是演示了单通道的情况,对于多通道的例子,请看 Code2 然后再看 Code3。Fn 2 :使用 Mat:at 函数原型 template inline _Tp& Mat:at() /其中参数有多个,也就是说 at 函数有多个重载返回值为 Mat 类型, Mat 有个索引的重载,也就是

11、符号的重载,用这个重载可以定位多通道数据,具体例如可以看下面代码。下面的代码把红色通道值大于128的颜色的置为白色,左边为原图,右边为处理过后的图。Code 2 :int main() Mat img = imread(lena.jpg); imshow(Lena Original, img);for (int row = 0; row img.rows; row+) for (int col = 0; col img.cols; col+) /* 注意 Mat:at 函数是个模板函数, 需要指明参数类型, 因为这图是具有红蓝绿三通道的图, 所以它的参数类型可以传递一个 Vec3b, 这是一个

12、存放 3 个 uchar 数据的 Vec(向量). 这里 提供了索引重载, 2表示的是返回第三个通道, 在这里是 Red 通道, 第一个通道(Blue)用0返回 */if(img.at(row, col)2 128) img.at(row, col) = Vec3b(255, 255, 255); imshow(Lena Modified, img); cvWaitKey();return 0;Output 2 :Code 3 :这段代码用的是 Fn1 的方式,效果和 Code 2 等价,不过是处理三通道数据而已:int main() Mat img = imread(lena.jpg); i

13、mshow(Lena Original, img);for (int row = 0; row img.rows; row+) for (int col = 0; col 128) /row, col像素的第 1 通道地址被 * 解析(blue通道) *(img.data + img.step0 * row + img.step1 * col) = 255;/row, col像素的第 2 通道地址被 * 解析(green通道), 关于elemSize1函数的更多描述请见 Fn1 里所列的博文 *(img.data + img.step0 * row + img.step1 * col + im

14、g.elemSize1() = 255;/row, col像素的第 3 通道地址被 * 解析(red通道) *(img.data + img.step0 * row + img.step1 * col + img.elemSize1() * 2) = 255; imshow(Lena Modified, img); cvWaitKey();return 0;Output 3 = Output 2Fn 3 :使用 Mat 的一个模板子类 Mat_ 的 ( ) 符号重载定位一个像素Code 4 :int main() Mat m(400, 400, CV_8UC3, Scalar(255, 255

15、, 255);/ m2 是 Mat_ 类型的, 因为 m 中元素的类型是 CV_8UC3, 可以用 Vec3b 存储 3 个通道的值/ 注意 Mat_ 这种写法是错误的, 因为 CV_8UC3 只是一个宏定义/ #define CV_8UC3 CV_MAKETYPE(CV_8U, 3) Mat_ m2 = m;/ for 循环画一个红色的实心圆for (int y = 0; y m.rows; y+) for (int * = 0; * m.rows; *+) if (pow(double(*-200), 2) + pow(double(y-200), 2) - 10000.0 0.) / M

16、at_ 模板类实现了对()的重载, 可以定位到一个像素 m2(*, y) = Vec3b(0, 0, 255); imshow(Image, m); cvWaitKey();return 0;Output 4 : 看上去怎么有点不爽Fn 4 :使用 Mat:ptr 模板函数Code 5 :int main() Mat m(400, 400, CV_8UC3, Scalar(226, 46, 166); imshow(Before, m);for (int row = 0; row m.rows; row+) if (row % 5 = 0) / data 是 uchar* 类型的, m.ptr(row) 返回第 row 行数据的首地址/ 需要注意的是该行数据是按顺序存放的,也就是对于一个 3 通道的 Mat, 一个像素有/ 有 3 个通道值, B,G,RB,G,RB,G,R. 所以一行长度为:/ sizeof(uchar) * m.cols * m.channels() 个字节 uchar* data = m.ptr(row);for (int col = 0; col m.cols; col+) datacol * 3 = 102; /第row行的第col个像素点的

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论