1、类 Mat 导言OpenCV c + + n 维稠密数组类类 CV_EXPORTS Matpublic:/ / 很多的方法./*!包括几位字段:-神奇的签名-连续性标志-深度(Note :应该是位深)-通道数*/int flags;(Note :目前还不知道 flags 做什么用的)/!数组的维数, = 2int dims ;/!行和列的数量或 (-1,-1) 此时数组已超过 2 维int rows,cols;/!指向数据的指针uchar *data ;/!指针的引用计数器 ;/ / 阵列指向用户分配的数据时,当指针为 NULLint * refcount ;/ / 其他成员.;Mat 类表示
2、一个 n 维的密集数值单通道或多通道数组。它可以用于存储实数或复数值的向量和矩阵、灰度或彩色图像、体素、向量场、点云、张量、直方图 (尽管较高维的直方图存储在 SparseMat 可能更好)。M 数组的数据布局是由阵列 M.step定义的,使元素的地址(i 0,。 iM.dims-1),其中 0 =M.stepi+1 (事实上,M.stepi =M.stepi+1*M.sizei+1)。这意味着 2 维矩阵是按行存储的,3 维矩阵是由平面存储,以此类推。M.stepM.dims-1 是最小的而且总是等于元素大小 M.elemSize()。因此,Mat 中的数据布局完全兼容 OpenCV 1.x
3、 中 CvMat、 IplImage、 CvMatND 类型。它也和标准工具包和 SDK,如Numpy(ndarray),Win32(独立设备位图) 等主流的密集数组类型相兼容,也就是说,与任何使用步进(或步长)来计算像素位置的阵列相兼容。由于这种兼容性,使用户分配的数据创建 Mat 头以及用 OpenCV 函数实时处理该头成为可能。有很多不同的方法,创建一个 Mat 的对象。下面列出了最常见的选项:使用 create(nrows,ncols,type)方法或类似的 Mat(nrows,ncols,type ,fillValue)构造函数。一个新的指定了大小和类型的数组被分配。type 和 c
4、vCreateMat 方法中的type 参数具有相同的含义。例如, CV_8UC1 是指一个 8 位单通道阵列,CV_32FC2 指 2 通道(复)浮点阵列,以此类推。/创建一个用 1+3j 填充的 7 x 7 复矩阵。Mat M(7,7,CV_32FC2,Scalar(1,3) ;/ /现在将 M 转换为 100 x 60 的 CV_8UC(15)的矩阵。/ / 旧内容将会被释放M.create(100,60,CV_8UC(15) ;这一章导言中指出,当当前的数组与指定的数组的形状或类型 create() 分配唯一的新数组时的形状或类型。创建多维数组:/ / 创建 100 x 100 x 1
5、00 8 位数组int sz = 100, 100, 100;Mat. bigCube (3,sz,CV_8U,Scalar:all(0) ;它将维度数(= 1)传递给 Mat 的构造函数,但列数设置为 1 时,创建数组将是 2 维的。因此,Mat:dims 始终是=2 的( 该数组为空时,也可以是 0)。使用的复制构造函数或赋值运算符可以是一个数组或右侧的表达式(请参阅下图)。正像在导言中指出的,数组赋值运算复杂度是 O(1)因为当你需要它的时候,它仅复制头和增加引用计数。Mat:clone() 方法可用于获取全(深)的副本数组。为另一个数组的一部分构建头。它可以是单个行、 单个列,几个行,
6、几个列,矩形区域(代数中称为较小值) 的数组或对角线。这种操作也是复杂度为 O(1),因为,新头引用相同的数据。实际上,您可以使用此特性修改该数组的一部分例如:/ /第 5 行,乘以 3,加到第 3 行,M.row(3) = M.row(3) + M.row (5) * 3 ;/ / 现在将第 7 列复制到第 1 列/ / M.col(1) = M.col(7) ;/ / 这个不能实现。Mat M1= M.col(1) ;M.col(7).copyTo(M1) ;/ / 创建一种新的 320 x 240 图像Mat img(Size(320,240),CV_8UC3) ;/ / 选择 ROI(
7、region of interest)Mat roi(img,Rect(10,10,100,100) ;/ / 填充 (0,255,0 ) 的 ROI (这是 RGB 空间中的绿色);/ / 320 x 240 原始图像将被修改。roi = Scalar(0,255,0) ;由于额外的 datastart 和 dataend 的成员,它们使得用 locateROI() 计算子数组在主容器数组中的相对的位置成为可能:Mat A = Mat:eye ( 10, 10, CV_32S);/ / 提取 A 的 1 (含)到 3 (不包含)列。Mat B = A(Range:all(),Range(1,
8、3 ) ;/ / 提取 B 的 5 (含)到 9 (不包含)行。/ /即 C A(Range(5,9 ),Range (1,3 )Mat C = B(Range(5,9),Range:all() ;Size size;Point ofs;C.locateROI ( size,ofs);/ / size 将变为 (width= 10, height= 10),ofs 会变为 (x = 1,y = 5)考虑到整个矩阵,如果您需要深层副本,使用子矩阵的 sclone() 方法的提取。为用户分配数据创建矩阵头。有利于执行下列操作:1. 使用 OpenCV 处理“外来“的数据(例如,当您执行 Direc
9、tShow *lter 或 gstreamer的 pro-cessing 模块,等等)。例如:void process_video_frame (const unsignedchar * pixels,int width,int height,int step)Mat img (width,height, CV_8UC3,pixels,step);GaussianBlur (img,img ,Size(7,7),1.5,1.5) ;2.快速初始化小矩阵和/或获取超快的元素的访问。double m3 3 = a,b,c,d,e,f g, h, i;Mat M = Mat(3,3,CV_64F,m
10、).inv() ;本例中用户分配数据的一些很常见情况是从 CvMat 和 IplImage 转换到 Mat。为达到此目的,有些特殊的构造函数以指向 CvMat 或 IplImage 和 ag可选参数指示是否数据复制。从 Mat 到 CvMat 或 IplImage 的后台转换是通过类型转换运算符 Mat:operator CvMat() const 和 Mat:operator IplImage()实现的。operators 不要复制数据。IplImage * img = cvLoadImage(“greatwave.jpg“,1) ;Mat mtx(img) ;/ / IplImage *-
11、 MatCvMat oldmat = mtx ;/ / Mat- CvMatCV_Assert (oldmat.cols = = img- width使用 MATLAB 样式数组初始值设定项 zeros()、 ones()、 eye(),例如:/ / 创建具双精度标识矩阵并将其添加到 M。M + = Mat:eye (M.rows,M.cols,CV_64F);使用逗号分隔的初始值设定项:/ / 创建 3 x 3 双精度恒等矩阵Mat M = (Mat_ (3,3) (3,3)(i,j) + = 1.f ;假定 M 一个双精度浮点型数组。有几个变体的不同方法来针对不同的维度数进行处理。如果您要
12、处理整行的二维数组,最有效的方式是获取该行的头指针然后只需使用普通的 C运算符:/ / 正矩阵元素之和计算/ / (假定 M 是一个双精度矩阵)double sum = 0;for (int i = 0 ;i (i) ;for (int j = 0; j (i) ;for (int j = 0; j it =M.begin (),it_end = M.end () ;for(; it! = it_end ; +it)sum+ = std:max (*it,0.);矩阵迭代器是随机存取的迭代器,所以他们可以被传递给任何 STL 算法,包括 std:sort()。矩阵表达式这是已经实现的可以组合在
13、任意复杂的表达式中的矩阵运算操作, (此处 A 、B 的表示矩阵 (Mat)、 s 表示标量(Scalar),alpha 为实数标量 (双精度型):加法、减法、求反: A + B + A-B、 A + s、 A-s、 s + A、 s-A、-A;缩放: A * 阿尔法每个元素乘法和除法: A.mul (B)、 A / B,alpha/A矩阵相乘: A * B大动脉转位: A.t() (指在)矩阵反演和伪反演,求解线性系统和最小二乘问题:A.inv(method) ( A-1) , A.inv(method)*B ( X: AX=B)比较: cmpop B、 cmpop alpha、 alpha
14、 cmpop A,其中 cmpop 是以下几种运算符之一: , =,= =,! =, () 构造函数将结果强制转换为适当的类型。Note:有些逗号分隔初始值设定项和一些其他的运算符可能需要显示调用 Mat();或Mat_();的构造函数来解决可能产生的歧义。以下是一些矩阵表达式的例子:/计算矩阵 A 的伪反演等价于 A.inv(DECOMP_SVD)SVD svd(A);Mat pinvA =svd.vt.t()*Mat:diag(1./svd.w)*svd.u.t();/计算莱文伯格-马夸特算法中的参数的新向量x -= (A.t()*A +lambda*Mat:eye(A.cols,A.co
15、ls,A.type().inv(DECOMP_CHOLESKY)*(A.t()*err);/用“Unsharp Mask”算法锐化图像Mat blurred; double sigma = 1, threshold =5, amount = 1;GaussianBlur(img, blurred, Size(), sigma,sigma);Mat lowConstrastMask = abs(img - blurred)explicit Mat:Mat(const VecNote:在当前实现中,下面的代码不会无法按预期的效果工作:Mat A ;.A.row(i) = A.row(j) ;/ /
16、不起作用发生这种情况是因为 A.row(i) 形成临时矩阵头进一步分配给另一个矩阵头。请记住,每个操作复杂度为 O(1),即没有复制任何数据。因此,如果你预期第 j 行被复制到第 i 行,那么上述赋值不成立。要做到这一点,应该把这种简单的赋值转换到表达式中或使用 Mat:copyTo() 方法:Mat A ;./ / 可行,但看上去有点目的不明确。A.row(i) = A.row(j) + 0;/ / 这是有点儿长,但这是推荐的方法。A.row(j).copyTo(A.row(i) ;Mat:col创建一个具有指定了矩阵头中列数这个参数的矩阵C+: Mat Mat:col(int j) con
17、st参数:j 一个 0 基(从 0 开始)的列索引该方法创建一个具有指定了矩阵头中列数这个参数的新矩阵并作为函数返回值。这是一种复杂度为 O(1)的操作,不用考虑矩阵的尺寸大小。新矩阵和原始矩阵共享一份基础数据。参看 Mat:row()说明信息。Mat:rowRange为指定的行 span 创建一个新的矩阵头。C+: Mat Mat:rowRange(int startrow, int endrow) constC+: Mat Mat:rowRange(const Range因此,目标矩阵会在必要的情况下重新分配尽管 m.copyTo(m) works awlessly,该函数并不处理源矩阵和
18、目标矩阵之间有重叠的部分的情况。当操作掩码指定以及上述的 Mat:create 重新分配矩阵,新分配的矩阵在数据复制到里面之前全都被初始化为。Mat:convertTo在缩放或不缩放的情况下转换为另一种数据类型。C+:void Mat:convertTo(OutputArray m,int rtype,double alpha=1,double beta=0)const参数:m 目标矩阵。如果它的尺寸和类型不正确,在操作之前会重新分配。rtype 要求是目标矩阵的类型,或者在当前通道数与源矩阵通道数相同的情况下的depth。如果 rtype 为负,目标矩阵与源矩阵类型相同。beta 可选的 d
19、elta 加到缩放值中去。该方法将源像素值转化为目标类型 saturate_cast 要放在最后以避免溢出m( x;y) = saturate_cast ( *( *this)( x;y) +)Mat:assignTo提供了一个 convertTo 的功能形式。C+: void Mat:assignTo(Mat.Mat pointMat = Mat(vec). /把向量转化成 Mat, 复杂度为 O(1)的运算reshape(1). / 从 Nx1 的 3 通道矩阵得出 Nx3 的单通道矩阵/同样是复杂度为 O(1)的运算t(); / 最后转置 Nx3 的矩阵/这个过程要复制所有的元素Mat:
20、t转置矩阵。.C+: MatExpr Mat:t() const该方法通过矩阵表达式(matrix expression)实现矩阵的转置 The method performs matrix transposition by means of matrix expressions. 它并未真正完成了转置但却返回一个临时的可以进一步用在更复杂的矩阵表达式中或赋给一个矩阵的转置矩阵对象:Mat A1 = A + Mat:eye(A.size(), A.type)*lambda;Mat C = A1.t()*A1; /计算(A + lambda*I)t * (A + lamda*I).Mat:inv
21、反转矩阵C+: MatExpr Mat:inv(int method=DECOMP_LU) const参数:method 反转矩阵的方法。有以下几种可能的值: DECOMP_LU 是 LU 分解一定不能是单数的。 DECOMP_CHOLESKY 是 Cholesky LLT 只适用于对称正矩阵的分解。该类型在处理大的矩阵时的速度是 LU 的两倍左右。 DECOMP_SVD 是 SVD 分解。如果矩阵是单数或甚至不是 2 维,函数就会计算伪反转矩阵。该方法执行矩阵的反转矩阵表达。这意味着该方法返回一个临时矩阵反转对象并可进一步用于更复杂的矩阵表达式的中或分配给一个矩阵。Mat:mul执行两个矩阵
22、按元素相乘或这两个矩阵的除法。C+: MatExpr Mat:mul(InputArray m, double scale=1) const参数:m 与 *this 具有相同类型和大小的矩阵,或矩阵表达式。scale 可选缩放系数。该方法返回一个用可选的缩放比率编码了每个元素的数组乘法的临时的对象。 注意:这不是一个对应“*” 运算符的简单的矩阵乘法。.例::Mat C = A.mul(5/B); / 等价于 divide(A, B, C, 5)Mat:cross计算 3 元素向量的一个叉乘积。C+: Mat Mat:cross(InputArray m) const参数:m 另一个叉乘操作对
23、象。该方法计算了两个元素向量的叉乘的积被操作向量必须是元素浮点型的具有相同形状和尺寸的向量。结果也是一语被操作对象的具有相同形状和大小的浮点型元素向量。Mat:dot计算两向量的点乘。C+: double Mat:dot(InputArray m) const参数:m 另一个点积操作对象。方法计算两个矩阵的点积。如果矩阵不单列或单行的向量,用顶部到底部从左到右扫描次序将它们视为 1 D 向量。这些向量必须具有相同的大小和类型。如果矩阵有多个通道,从所有通道得到的点积会被加在一起。Mat:zeros返回指定的大小和类型的零数组。C+: static MatExpr Mat:zeros(int r
24、ows, int cols, int type)C+: static MatExpr Mat:zeros(Size size, int type)C+: static MatExpr Mat:zeros(int ndims, const int* sizes, int type)参数ndims 数组的维数。rows行数。cols 列数。size替代矩阵大小规格 Size(cols, rows)的方法。sizes 指定数组的形状的整数数组。type 创建的矩阵的类型。该方法返回一个 Matlab 式的零数组初始值设定项。它可以用于快速形成一个常数数组作为函数参数,作为矩阵的表达式或矩阵初始值设定
25、项的一部分。Mat A;A = Mat:zeros (3,3 ,CV_32F);在上面的示例中,只要 A 不是 3 x 3 浮点矩阵它就会被分配新的矩阵。否则为现有的矩阵 A 填充零。Mat:ones返回一个指定的大小和类型的全为 1 的数组。C+: static MatExpr Mat:ones(int rows, int cols, int type)C+: static MatExpr Mat:ones(Size size, int type)C+: static MatExpr Mat:ones(int ndims, const int* sizes, int type)参数:ndim
26、s 数组的维数。rows 行数。.cols 列数。size 替代矩阵大小规格 Size(cols, rows)的方法。sizes 指定数组的形状的整数数组。type 创建的矩阵的类型。该方法返回一个 Matlab 样式 1 的数组初始值设定项,类似 Mat:zeros()。请注意,这种方法中你可以使用任意一个值和 Matlab 语法初始化数组如下:Mat A = Mat:ones (100,100,CV_8U) * 3 ;/ / 使 100 x 100 矩阵里充满 3。上述操作不会形成一个 100 x 100 1 的矩阵,然后乘以 3。相反,它只是记住缩放因子(在本例中 3)在实际调用矩阵初始值设定项时使用它。