来自 OpenCV—基本矩阵操作与示例
这博主写得挺详细的。
OpenCV 中的矩阵操作非常重要。 要熟悉起来!
学习该博主的基本示例:
建立矩阵必须要指定矩阵储存的数据类型, 图像处理中常用的几种数据类型如下 :
CV_8UC1 // 8 bit 无符号单通道
CV_8UC3 // 8 bit 无符号 3 通道
CV_8UC4 //
CV_32FC1 // 32 bit 浮点型单通道
CV_32FC1 // 32 bit 浮点型 3 通道
可以通过载入图像来创建 Mat 类型矩阵, 也可以直接手动创建矩阵, 基本方法是指定矩阵尺寸和数据类型:
// 1.2 基本方法
cv::Mat a(cv::Size(5,5), CV_8UC1); // 单通道
cv::Mat b = cv::Mat(cv::Size(5,5), CV_8UC3); // 3 通道每个矩阵元素包含 3 个 uchar 值
cout << "a = " << endl << a << endl << endl;
cout << "b = " << endl << b << endl << endl;
system("color A");
system("pause");
3 通道矩阵中, 一个矩阵元素包括 3 个变量。
可以看到上面的矩阵数值并没有初始化, 避免这种情况, 使用 Mat 类的几种初始化创建矩阵的方法 :
// 1.3 初始化方法
cv::Mat m_Zeros = cv::Mat::zeros(cv::Size(5,5), CV_8SC1); // 全零矩阵
cv::Mat m_One = cv::Mat::ones(cv::Size(5,5), CV_8UC1); // 全 1 矩阵
cv::Mat m_Eye = cv::Mat::eye(cv::Size(5,5), CV_32FC1); // 对角线为 1 的矩阵
cout << "m_Zeros = " << endl << m_Zeros << endl << endl ;
cout << "m_One = " << endl << m_One << endl << endl ;
cout << "m_Eye = " << endl << m_Eye << endl << endl ;
system("color A");
system("pause");
OpenCV 的 Mat 类允许所有的矩阵运算。
使用 +
或者 -
符号进行矩阵加减运算。
// 2.2 矩阵加减法
Mat a = Mat::eye(Size(3,2), CV_32F);
Mat b = Mat::ones(Size(3,2), CV_32F);
Mat c = a+b;
Mat d = a-b;
cout << "a" << endl << a << endl << endl ;
cout << "b" << endl << b << endl << endl ;
cout << "c = a+b" << endl << c << endl << endl ;
cout << "d = a-b " << endl << d << endl << endl ;
system("color A");
system("pause");
矩阵与矩阵相乘, 必须满足矩阵相乘的行列数对应规则。
// 2.3 矩阵乘法
Mat m1 = Mat::eye(2,3, CV_32F);
Mat m2 = Mat::ones(3,2, CV_32F);
cout << "m1 =" << endl << m1 << endl << endl;
cout << "m2 =" << endl << m2 << endl << endl;
// Scalar by Matrix
cout << "\nm1.*2 = \n" << m1*2 << endl;
// Matrix per element multiplication
cout << "\n(m1+m2).*(m1+m3) = \n" << (m1+1).mul(m1+3) << endl;
// Matrix mutiplication
cout << "\nm1*m2 = \n" << m1*m2 << endl;
system("color A");
system("pause");
矩阵转置是将矩阵的行与列顺序对调, 将第几行转变为第几列。OpenCV 通过 Mat 类的 t()
函数实现。
// 2.4 矩阵转置
Mat m1 = Mat::eye(2,3, CV_32F);
Mat m1_t = m1.t();
cout << "m1 = " << endl << m1 << endl << endl;
cout << "m1t = " << endl << m1_t << endl << endl;
system("color A");
system("pause");
逆矩阵在一些算法中经常出现, 在 OpenCV 中通过 Mat 类的 inv()
方法实现。
// 2.5 求逆矩阵
Mat me = Mat::eye(5,5, CV_32FC1);
Mat meinv = me.inv();
cout << "me = "<< endl << " "<< me << endl << endl;
cout << "meinv = "<< endl << " " << meinv << endl << endl;
// 单位矩阵的逆就是本身.
system("color A");
system("pause");
计算物体的像素或者面积常需要用到矩阵中的非零元素个数, OpenCV 中使用 countNonZero() 函数实现。
Mat me = Mat::eye(6,6, CV_32FC1);
int nonZerosNum = countNonZero(me); // me 为输入矩阵或者图像
cout << "me = " << endl << " " << me << endl << endl;
cout << "me 中非零元素的个数 =" << nonZerosNum << endl << endl;
system("color A");
system("pause");
OpenCV 提供了矩阵均值和标准差计算功能, 可以使用 meanStdDev(src, mean, stddev)
函数实现
Mat me = Mat::eye(5,5, CV_32FC1);
Mat mean;
Mat stddev;
meanStdDev(me, mean, stddev); // me 为定义的对角矩阵
cout << "me = " << endl << " " << me << endl << endl;
cout << "mean = " << endl << " " << mean << endl << endl;
cout << "stddev = " << endl << " " << stddev << endl << endl;
system("color A");
system("pause");
单通道运行结果:
如果 src 是多通道图像或者多维矩阵, 则函数分别计算不同通道的均值与标准差, 因此返回值 mean 和 stddev 为对应维度的向量。
Mat mean3;
Mat stddev3;
Mat m3(cv::Size(5,5), CV_8UC3, Scalar(255,200,100));
cout << "m3 = " << endl << " " << m3 <cout << "mean3 = " << endl << " " << mean3 << endl << endl;
cout << "stddev3 = " << endl << " " << stddev3 << endl << endl;
system("color A");
system("pause");
多通道运行结果 :
求输入矩阵的全局最大值最小值及其位置, 可使用函数:
void minMaxLoc(InputArray src, CV_OUT double* minVal,
CV_OUT double* maxVal=0, CV_OUT Point* minLoc=0,
CV_OUT Point* maxLoc=0, InputArray mask=noArray());
参数 :
src
, 输入单通道矩阵 (图像).minVal
, 指向最小值的指针, 如果未指定则使用 NULL.maxVal
, 指向最大值的指针, 如果未指定则使用 NULL.minLoc
, 指向最小值位置 (2 维情况) 的指针, 如果未指定则使用 NULL.mask
, 可选的蒙版, 用于选择待处理区域.// 求极值: 最大值、最小值及其位置
Mat img = imread("Lena.jpg",0);
imshow("original image", img);
ofstream fout("lena1.txt"); // 保存数据的文件
double minVal = 0, maxVal = 0;
cv::Point minPt, maxPt;
minMaxLoc(img, &minVal, &maxVal, &minPt, &maxPt);
cout << "min value = " << minVal << endl;
cout << "max value = " << maxVal << endl;
cout << "minPt = " << minPt << endl;
cout << "maxPt = " << maxPt << endl;
fout << "min value = " << minVal << endl;
fout << "max value = " << maxVal << endl;
fout << "minPt = " << minPt << endl;
fout << "maxPt = " << maxPt <10, minPt.y - 10, 20, 20);
Rect rectMax(maxPt.x - 10, maxPt.y - 10, 20, 20);
rectangle(img, rectMin, cv::Scalar(200,2));
rectangle(img, rectMax, cv::Scalar(255,2));
imshow("img with min max loction", img);
system("color A");
waitKey(0);
其他详情矩阵操作可以参考 OpenCV Operations on Arrays