【和春笋一起学C++】OpenCV中数组和指针运用实例

**前言:**前面学习了数组和指针在C++中的处理原理,本文通过自己编写一个图像处理的函数实例来加深对数组和指针的理解。为什么是图像处理呢,因为图像数据是一个二维矩阵,相当于一个二维数组,前面学习了一维数组,现在看看指针是怎么来处理二维数组的。

目录

遍历OpenCV中图像像素方法一

遍历OpenCV中图像像素方法二

总结


**正文:**我们首先使用opencv图像处理库读入一张图像,然后对图像进行二值化(opencv库中已经集成了此功能,此处自己编写仅仅为了巩固指针和数组的知识)。

遍历OpenCV中图像像素方法一

代码如下:

cpp 复制代码
#include <iostream>
#include <cstring>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;

int main()
{
	Mat srcImage = imread("test.jpg");
	Mat srcGray;
	cvtColor(srcImage, srcGray, COLOR_BGR2GRAY);
	Mat outputImage = srcGray.clone();
	int rows = outputImage.rows;
	int cols = outputImage.cols*outputImage.channels();
	int threshold = 50;
	for (int i = 0; i < rows; i++)
	{
		uchar* data = outputImage.ptr<uchar>(i);///获取第i行的首地址
		for (int j = 0; j < cols; j++)
		{
			if (data[j] < threshold)
			{
				data[j] = 255;
			}
			else
			{
				data[j] = 0;
			}
		}
	}
	imshow("src", srcImage);
	imshow("binary", outputImage);
	waitKey();
	return 0;
}

该方法中最重要的是这条语句:uchar* data = outputImage.ptr<uchar>(i);Mat类中的ptr函数返回的是某一行的首地址,ptr是一个模板函数,代码中它返回的是第i行的首地址。

第一个循环体内获取每行的首地址,第二个循环体内获取每行的每个像素值,并做阈值划分。前面文章《使用new创建动态数组》讲过,访问数组中的元素既可以用指针访问,也可以用数组的方括号表示法来访问,此处用的是方括号表示法,data[j]代表的是第i行中的第(j+1)个像素值,通过双层循环对图像中每一行的每个像素值做阈值划分,就完成了图像的二值化。

也可以通过指针的方式来访问图像中的每个像素值,将上面代码中的循环体换成以下代码,运行的结果是一样的。

cpp 复制代码
for (int i = 0; i < rows; i++)
	{
		uchar* data = outputImage.ptr<uchar>(i);///获取第i行的首地址
		for (int j = 0; j < cols; j++)
		{
			if ((*(data+j)) < threshold)
			{
				*(data + j) = 255;
			}
			else
			{
				*(data + j) = 0;
			}
		}
	}

遍历OpenCV中图像像素方法二

方法一中通过获取每行的首地址来访问每行的每个像素值。也可以直接通过二维数组的整个数组的首地址来访问每个像素值。

其实,当opencv读入一张图像时,图像数据(每个像素的像素值)在内存中的存储是连续的。当opencv读入一张图片时,计算机将图像中每行的像素值一行一行的连接起来,存储在一块连续的内存块中。假如一张图片大小为500*600(row=500,col=600),且为单通道图像,则第600个数据为图像第一行的最后一个像素值,第601个数据为图像第二行的第1个像素值。

在opencv中可以通过Mat类中的函数isContinuous来判断数据是否连续存储。图像数据的连续存储有助于提升图像的扫描速度,相当于将图像中的每行一行一行的连接起来,形成一个长行。此时,遍历每个像素只需要一个指针或一个一维数组。

代码如下:

cpp 复制代码
#include <iostream>
#include <cstring>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;

int main()
{
	Mat srcImage = imread("test.jpg");
	Mat srcGray;
	cvtColor(srcImage, srcGray, COLOR_BGR2GRAY);
	Mat outputImage = srcGray.clone();
	int rows = outputImage.rows;
	int cols = outputImage.cols*outputImage.channels();
	int threshold = 50;
    if (outputImage.isContinuous() == false)
	{
		cout << "图像数据非连续存储" << endl;
		return 0;
	}
	else
	{
		cout << "图像数据为连续存储" << endl;
	}
	uchar* data = outputImage.data;
	for (int i = 0; i < rows; i++)
	{
		for (int j = 0; j < cols; j++)
		{
			if (data[i*cols+j] < threshold)
			{
				data[i*cols + j] = 255;
			}
			else
			{
				data[i*cols + j] = 0;
			}
		}
	}
	imshow("src", srcImage);
	imshow("binary", outputImage);
	waitKey();
	return 0;
}

方法二中最重要的是这条语句:uchar* data = outputImage.data; Mat类中的data指针指向的就是图像数据二维数组的首地址,该方法用的是数组的方括号表示法来访问每个像素。同理,循环体中也可以换成指针的方式来访问每个像素。循环体中的代码换成指针的方式如下:

cpp 复制代码
for (int i = 0; i < rows; i++)
	{
		for (int j = 0; j < cols; j++)
		{
			if (*(data+i*cols + j) < threshold)
			{
				*(data + i * cols + j) = 255;
			}
			else
			{
				*(data + i * cols + j) = 0;
			}
		}
	}

总结

opencv中遍历图像像素的方法有十几种,本文通过4种遍历图像像素的方式编写了一个简单的图像二值化函数,既巩固了指针和数组的知识,又掌握了遍历opencv中图像像素的方法。

相关推荐
Q741_14723 分钟前
每日一题 力扣 2840. 判断通过操作能否让字符串相等 II 力扣 2839. 判断通过操作能否让字符串相等 I 找规律 字符串 C++ 题解
c++·算法·leetcode·力扣·数组·找规律
格林威2 小时前
Baumer相机铝型材表面划伤长度测量:实现损伤量化评估的 5 个关键技术,附 OpenCV+Halcon 实战代码!
开发语言·人工智能·数码相机·opencv·计算机视觉·c#·工业相机
格林威3 小时前
Baumer相机铝箔表面针孔检测:提升包装阻隔性的 7 个核心策略,附 OpenCV+Halcon 实战代码!
开发语言·人工智能·数码相机·opencv·计算机视觉·c#·工业相机
不懒不懒4 小时前
【OpenCV 计算机视觉四大核心实战:从背景建模到目标跟踪】
人工智能·python·opencv·机器学习·计算机视觉
格林威4 小时前
Baumer相机芯片引脚共面性检测:保障电子装配精度的 5 个实用方案,附 OpenCV+Halcon 实战代码!
开发语言·人工智能·opencv·计算机视觉·c#·视觉检测·工业相机
格林威5 小时前
Baumer相机金属冲压件毛刺高度测量:量化去毛刺效果的 5 个核心方法,附 OpenCV+Halcon 实战代码!
人工智能·opencv·计算机视觉·c#·视觉检测·机器视觉·工业相机
我材不敲代码18 小时前
OpenCV实战:全自动答题卡识别与评分系统
人工智能·opencv·计算机视觉
li三河21 小时前
opencv利用freetype写中文
人工智能·opencv·计算机视觉
AI科技星1 天前
基于四维时空光速不变公设的量子几何与量子力学本质全维度推导验证
开发语言·人工智能·opencv·计算机视觉·数学建模·r语言
纤纡.1 天前
实战 OpenCV:从文档扫描到目标追踪,四大核心场景全解析
人工智能·opencv·计算机视觉