矩阵运算提速——玩转opencv::Mat
介绍:用Eigen或opencv::Mat进行矩阵的运算,比用cpp的vector或vector进行矩阵运算要快吗?
使用 Eigen 或 OpenCV 的 cv::Mat
进行矩阵运算通常比使用 std::vector<int>
或 std::vector<double>
更快。这主要有以下几个原因:
-
优化的底层实现:Eigen 和 OpenCV 都是为高效的矩阵运算而设计的库,内部实现了许多优化,例如使用 SIMD(单指令多数据)指令、并行计算等。这些优化可以显著提高计算速度。
-
内存布局:Eigen 和 OpenCV 在内存布局上进行了优化,通常使用连续的内存块来存储矩阵数据,这样可以提高缓存命中率,从而加快访问速度。
-
专用的算法:这些库提供了许多专用的算法和函数,能够高效地执行常见的矩阵运算(如乘法、求逆、特征值分解等),而使用
std::vector
进行这些操作时,通常需要手动实现算法,效率较低。 -
类型安全和表达能力:Eigen 提供了强类型的矩阵和向量类,能够在编译时进行更多的检查,减少运行时错误,同时也使得代码更易于阅读和维护。
因此,如果你的应用涉及大量的矩阵运算,建议使用 Eigen 或 OpenCV 这样的专用库,而不是直接使用 std::vector
。
构造
link1
索引
访问Mat的通道数
#include <opencv2/opencv.hpp>
#include <iostream>
int main() {
// 创建一个3通道的彩色图像(例如,BGR格式)
cv::Mat colorImage = cv::Mat::zeros(100, 100, CV_8UC3);
// 创建一个单通道的灰度图像
cv::Mat grayImage = cv::Mat::zeros(100, 100, CV_8UC1);
// 获取通道数
int colorChannels = colorImage.channels();
int grayChannels = grayImage.channels();
// 输出通道数
std::cout << "彩色图像的通道数: " << colorChannels << std::endl; // 应该输出3
std::cout << "灰度图像的通道数: " << grayChannels << std::endl; // 应该输出1
return 0;
}
定义4维Mat
#include <opencv2/opencv.hpp>
#include <iostream>
int main() {
// 定义4维矩阵,大小为2x3x4x5,数据类型为CV_32F(32位浮点数)
cv::Mat mat4D(2, new int[4]{3, 4, 5}, CV_32F);
// 填充矩阵
for (int i = 0; i < 2; ++i) {
for (int j = 0; j < 3; ++j) {
for (int k = 0; k < 4; ++k) {
for (int l = 0; l < 5; ++l) {
mat4D.at<float>(i, j, k, l) = static_cast<float>(i * 1000 + j * 100 + k * 10 + l);
}
}
}
}
// 输出矩阵的形状和内容
std::cout << "4维矩阵的大小: " << mat4D.size << std::endl;
std::cout << "4维矩阵的内容:" << std::endl;
for (int i = 0; i < 2; ++i) {
for (int j = 0; j < 3; ++j) {
for (int k = 0; k < 4; ++k) {
for (int l = 0; l < 5; ++l) {
std::cout << mat4D.at<float>(i, j, k, l) << " ";
}
std::cout << std::endl;
}
std::cout << std::endl;
}
}
// 释放动态分配的内存
delete[] mat4D.size;
return 0;
}
取某一行
#include <opencv2/opencv.hpp>
#include <iostream>
int main() {
// 创建一个 3x3 的矩阵
cv::Mat mat = (cv::Mat_<float>(3, 3) << 1, 2, 3,
4, 5, 6,
7, 8, 9);
// 获取第 3 行(索引为 2)的所有元素
cv::Mat thirdRow = mat.row(2); // 行索引从 0 开始
// 输出结果
std::cout << "第三行的元素是:" << std::endl;
std::cout << thirdRow << std::endl;
return 0;
}
提取块
#include <opencv2/opencv.hpp>
#include <iostream>
int main() {
// 创建一个 5x5 的矩阵
cv::Mat mat = (cv::Mat_<float>(5, 5) << 1, 2, 3, 4, 5,
6, 7, 8, 9, 10,
11, 12, 13, 14, 15,
16, 17, 18, 19, 20,
21, 22, 23, 24, 25);
// 定义要提取的块的起始位置和大小
int startRow = 1; // 起始行索引
int startCol = 1; // 起始列索引
int blockRows = 3; // 块的行数
int blockCols = 3; // 块的列数
// 提取块
cv::Mat block = mat(cv::Range(startRow, startRow + blockRows), cv::Range(startCol, startCol + blockCols));
// 输出结果
std::cout << "提取的块是:" << std::endl;
std::cout << block << std::endl;
return 0;
}
访问某行某列的元素
在C++中,使用OpenCV库的cv::Mat类来表示图像或矩阵。要访问cv::Mat中的特定行和列的元素,可以使用at(row, col)方法,其中type是元素的数据类型。
#include <iostream>
#include <opencv2/opencv.hpp>
int main() {
// 创建一个3x3的矩阵,类型为CV_8UC1(单通道8位无符号整数)
cv::Mat mat = (cv::Mat_<uchar>(3, 3) << 1, 2, 3,
4, 5, 6,
7, 8, 9);
// 输出整个矩阵
std::cout << "矩阵内容:\n" << mat << std::endl;
// 访问特定行和列的元素
int row = 1; // 第二行(索引从0开始)
int col = 2; // 第三列(索引从0开始)
// 使用at方法访问元素
uchar value = mat.at<uchar>(row, col);
std::cout << "元素在 (" << row << ", " << col << ") 的值: " << (int)value << std::endl;
// 修改特定行和列的元素
mat.at<uchar>(row, col) = 10;
std::cout << "修改后的矩阵内容:\n" << mat << std::endl;
return 0;
}
示例输出:
矩阵内容:
[1, 2, 3;
4, 5, 6;
7, 8, 9]
元素在 (1, 2) 的值: 6
修改后的矩阵内容:
[1, 2, 3;
4, 5, 10;
7, 8, 9]
运算
矩阵乘法和Hamornoid积
#include <iostream>
#include <opencv2/opencv.hpp>
int main() {
// 创建两个相同维度的矩阵
cv::Mat mat1 = (cv::Mat_<float>(2, 2) << 1, 2,
3, 4);
cv::Mat mat2 = (cv::Mat_<float>(2, 2) << 5, 6,
7, 8);
// 输出原始矩阵
std::cout << "矩阵1:\n" << mat1 << std::endl;
std::cout << "矩阵2:\n" << mat2 << std::endl;
// 方法1:使用 cv::multiply
cv::Mat result1;
cv::multiply(mat1, mat2, result1);
std::cout << "矩阵乘法 (使用 cv::multiply):\n" << result1 << std::endl;
// 方法2:使用 cv::mul
cv::Mat result2 = mat1.mul(mat2); // 也可以使用 result2 = mat1 * mat2;
std::cout << "矩阵乘法 (使用运算符 *):\n" << result2 << std::endl;
// 方法3:使用运算符 *
cv::Mat result3 = mat1*mat2; // 也可以使用 result2 = mat1 * mat2;
std::cout << "逐元素相乘结果 (使用运算符 *):\n" << result3 << std::endl;
return 0;
}
类型一致
- opencv::Mat之间的加减乘除必须保证类型相同
- 比如cv::divide()运算
cv::divide(new_cen, cen_index[cen_id].size(), new_cen);
- opencv::Mat的类型有如下几种:
* 可在初始化时指定
Mat new_cen=Mat::zeros(1,128,CV_32FC1);
opencv::Mat是继承Eigen的实现吗?
cv::Mat
并不是继承自Eigen库的实现。OpenCV和Eigen是两个独立的库,它们各自有自己的数据结构和实现方式。
OpenCV的cv::Mat
:
cv::Mat
是OpenCV中用于表示图像和矩阵的主要数据结构。它提供了多种功能,包括图像处理、矩阵运算、图像读取和写入等。cv::Mat
支持多维数组,能够处理不同的数据类型(如8位无符号整数、浮点数等),并且可以表示单通道和多通道图像(如RGB图像)。- OpenCV的设计目标是高效处理图像和视频数据,特别是在计算机视觉和图像处理领域。
Eigen库:
- Eigen是一个用于线性代数的C++模板库,提供了矩阵和向量的高效操作。它专注于数学运算,特别是线性代数、几何变换和数值计算。
- Eigen的设计目标是提供高性能的数学运算,同时保持易用性和灵活性。
结论:
虽然OpenCV和Eigen都涉及矩阵和线性代数的概念,但它们是独立的库,cv::Mat
并不继承自Eigen的实现。实际上,OpenCV和Eigen可以一起使用,开发者可以在OpenCV中处理图像数据,然后使用Eigen进行更复杂的数学运算,或者反之亦然。两者之间的结合可以利用各自的优势。