第一种:at方法遍历
at方法遍历 单通道
#include<iostream>
#include<opencv2\opencv.hpp>
using namespace std;
using namespace cv;
int main() {
cv::Mat a = (cv::Mat_<uchar>(4, 4) << 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16);
cout << a << endl;
cout << endl;
//at方法遍历并修改,单通道
for (int i = 0; i < a.rows; i++) {
for (int j = 0; j < a.cols; j++) {//j每次移动代表一个像素点的距离,即每次移动一个通道的距离
a.at<uchar>(i, j) = i + j;
}
}
//
cout << a << endl;
return 0;
}
运行结果
at方法遍历 多通道
(下面这种方式,既能遍历,又能修改)
#include<iostream>
#include<opencv2\opencv.hpp>
using namespace std;
using namespace cv;
int main() {
cv::Mat a(4, 4, CV_8UC3, cv::Scalar(1, 2, 3));
cout << a << endl;
cout << endl;
//at方法遍历遍历并修改,三通道
for (int i = 0; i < a.rows; i++) {
for (int j = 0; j < a.cols; j++) {//j每次移动代表一个像素点的距离,即每次移动三个通道的距离
cv::Vec3b temp_channel_array;//temp_channel_array临时的通道数组
temp_channel_array[0] = 4;
temp_channel_array[1] = 5;
temp_channel_array[2] = 6;
a.at<cv::Vec3b>(i, j) = temp_channel_array;
}
}
cout << a << endl;
return 0;
}
运行结果:
对比程序:(下面这种方式,只能遍历,不能修改)
#include<iostream>
#include<opencv2\opencv.hpp>
using namespace std;
using namespace cv;
int main() {
cv::Mat a(4, 4, CV_8UC3, cv::Scalar(1, 2, 3));
cout << a << endl;
cout << endl;
//at方法遍历遍历,三通道
for (int i = 0; i < a.rows; i++) {
for (int j = 0; j < a.cols; j++) {//j每次移动代表一个像素点的距离,即每次移动三个通道的距离
cv::Vec3b temp_channel_array= a.at<cv::Vec3b>(i, j);//temp_channel_array临时的通道数组
temp_channel_array[0] = 4;
temp_channel_array[1] = 5;
temp_channel_array[2] = 6;
}
}
cout << a << endl;
return 0;
}
运行结果
第二种:指针ptr方法 遍历(两种写法)
第一种写法
#include<iostream>
#include<opencv2\opencv.hpp>
using namespace std;
using namespace cv;
int main() {
cv::Mat a(4, 4, CV_8UC3, cv::Scalar(1, 2, 3));
cout << a << endl;
cout << endl;
//指针ptr方法遍历遍历,三通道
for (int i = 0; i < a.rows; i++) {
uchar* ptr = a.ptr<uchar>(i);//ptr表示:指向第i行中第一个像素点中第一个通道的指针?
for (int j = 0; j < a.cols; j++) {
*(ptr+3*j+0) = 4;//(ptr+3*j+0)表示:指向第i行第3*j列像素点中第一个通道的指针?
*(ptr+3*j+1) = 5;//(ptr+3*j+1)表示:指向第i行第3*j列像素点中第二个通道的指针?
*(ptr+3*j+2) = 6;//(ptr+3*j+2)表示:指向第i行第3*j列像素点中第三个通道的指针?
}
}
cout << a << endl;
return 0;
}
运行结果
更一般的写法:把里面的3改成矩阵的通道数 a.channels(),如下:
#include<iostream>
#include<opencv2\opencv.hpp>
using namespace std;
using namespace cv;
int main() {
cv::Mat a(4, 4, CV_8UC3, cv::Scalar(1, 2, 3));
cout << a << endl;
cout << endl;
//指针ptr方法遍历遍历,三通道
for (int i = 0; i < a.rows; i++) {
uchar* ptr = a.ptr<uchar>(i);//ptr表示:指向第i行中第一个像素点中第一个通道的指针?
for (int j = 0; j < a.cols; j++) {
*(ptr + a.channels()*j + 0) = 4;//(ptr + a.channels()*j + 0)表示:指向第i行第a.channels()*j列像素点中第一个通道的指针?
*(ptr + a.channels()*j + 1) = 5;//(ptr + a.channels()*j + 1)表示:指向第i行第a.channels()*j列像素点中第二个通道的指针?
*(ptr + a.channels()*j + 2) = 6;//(ptr + a.channels()*j + 2)表示:指向第i行第a.channels()*j列像素点中第三个通道的指针?
}
}
cout << a << endl;
return 0;
}
运行结果
上面这种写法,是能看到目前这个指针 ptr + a.channels()*j + 0 所指向的是位于第几行第几列中的第几个通道的,下面的写法就不具备这种优势。
第二种写法
#include<iostream>
#include<opencv2\opencv.hpp>
using namespace std;
using namespace cv;
int main() {
cv::Mat a(4, 4, CV_8UC3, cv::Scalar(1, 2, 3));
cout << a << endl;
cout << endl;
//指针ptr方法遍历遍历,三通道
for (int i = 0; i < a.rows; i++) {
uchar* ptr = a.ptr<uchar>(i);//ptr表示:指向第i行中第一个像素点中第一个通道的指针?
for (int j = 0; j < a.cols*a.channels(); j++) {//a.cols*a.channels()表示:矩阵中,第i行中所有的通道数。ptr指针每次仅移动一个通道的距离
*(ptr + j) = 4;
//ptr[j] = 4;//或者这样也行,等价的写法
}
}
cout << a << endl;
return 0;
}
运行结果
上面这种写法,感觉是看不到目前这个指针 ptr+j 所指向的是位于第几行第几列中的第几个通道的,感觉不是很明了。
第三种:使用迭代器遍历
暂无文章来源地址https://www.toymoban.com/news/detail-534250.html文章来源:https://www.toymoban.com/news/detail-534250.html
第四种:直接使用 矩阵元素的地址定位 方式遍历
暂无
到了这里,关于OpenCV中cv::Mat矩阵的四种遍历的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!