C++ OpenCV实战之形状识别

 更新时间:2022年07月09日 10:40:36   作者:Zero___Chen  
本案例通过使用OpenCV中的approxPolyDP进行多边形近似,进而进行基础形状识别(圆、三角形、矩形、星形…),快跟随小编一起动手尝试一下

前言

本案例通过使用OpenCV中的approxPolyDP进行多边形近似,进而进行基础形状识别(圆、三角形、矩形、星形…)。下面就一起来看看具体是如何实现的吧。

一、图像预处理

原图如图所示:

首先第一步先进行图像预处理,得到二值图像。

	Mat gray;
	cvtColor(src, gray, COLOR_BGR2GRAY);

	Mat gaussian;
	GaussianBlur(gray, gaussian, Size(3, 3), 0);

	Mat thresh;
	threshold(gaussian, thresh, 0, 255, THRESH_BINARY_INV | THRESH_OTSU);

结果如图所示。接下来,需要对此二值图像进行轮廓提取,进而识别物体形状。

二、形状识别

本案例使用approxPolyDP进行形状识别,关于approxPolyDP OpenCV给出的定义是:

void approxPolyDP(InputArray curve, OutputArray approxCurve, double epsilon, bool closed);

  • curve:表示输入轮廓点集,可以是 vector 或 Mat 类型。
  • approxCurve:多边形逼近结果,存储在approxCurve数组中。curve和approxCurve应该属于同一类型。
  • epsilon:表示逼近准确度,你允许在原多边形和最终拟合的多边形之间存在的最大偏差。一般以其周长的百分比进行近似。
  • closed:指明curve中的一系列点是否是一个闭合的多边形。若设为true,则认为曲线是闭合的。

我们通过统计多边形的“边”数来识别物体形状。

三、源码

#include<iostream>
#include<opencv2/opencv.hpp>
using namespace std;
using namespace cv;

//基础几何形状识别
bool Pattern_Recognition(Mat& src)
{
	//图像预处理
	Mat gray;
	cvtColor(src, gray, COLOR_BGR2GRAY);

	Mat gaussian;
	GaussianBlur(gray, gaussian, Size(3, 3), 0);

	Mat thresh;
	threshold(gaussian, thresh, 0, 255, THRESH_BINARY_INV | THRESH_OTSU);

	//轮廓查找
	vector<vector<Point>>contours;//轮廓点集
	findContours(thresh, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE);
	vector<vector<Point>>conPoly(contours.size());//多边形逼近结果,与轮廓一一对应

	for (int i = 0; i < contours.size(); i++)
	{
		double area = contourArea(contours[i]); //轮廓面积

		if (area > 1000)
		{
			Rect rect = boundingRect(contours[i]);//外界矩形

			double ratio = double(rect.width) / double(rect.height);//长宽比

			double peri = arcLength(contours[i], true);//周长

			approxPolyDP(contours[i], conPoly[i], 0.02 * peri, true);//多边形近似

			int objSize = conPoly[i].size();//折线数--通过判断轮廓有几条边来识别图形

			string objName;
			Scalar color;
			if (objSize == 3)
			{
				objName = "Triangle";//三角形
				color = Scalar(0, 0, 255);
			}
			if (objSize == 4)
			{
				//通过长宽比判断正方形/长方形
				if (ratio > 0.99 && ratio < 1.01)
				{
					objName = "Square";//正方形
					color = Scalar(0, 255, 255);
				}
				else
				{
					objName = "Rectangle";//长方形
					color = Scalar(0, 255, 0);
				}
			}
			if (objSize == 8)
			{
				objName = "Circle";//圆形
				color = Scalar(255, 255, 0);
			}
			if (objSize == 10)
			{
				objName = "Star";//星形
				color = Scalar(255, 0, 255);
			}

			//效果绘制
			rectangle(src, rect, color, 2);
			putText(src, objName, rect.tl(), FONT_HERSHEY_SIMPLEX, 1, color, 2);
		}
	}

	return true;
}

int main()
{
	Mat src = imread("src.jpeg");
	if (src.empty())
	{
		cout << "No Image!" << endl;
		system("pause");
		return -1;
	}
	
	if (!Pattern_Recognition(src))return false;

	namedWindow("test", WINDOW_NORMAL);
	imshow("test", src);
	waitKey(0);
	system("pause");
	return 0;
}

四、结果显示

总结

本文使用OpenCV C++ 进行基础形状识别,其实原理很简单,主要操作有以下几点。

1、图像预处理

2、物体轮廓提取

3、使用approxPolyDP进行多边形近似,进而统计出该物体的“边”数,从而识别出物体形状。

到此这篇关于C++ OpenCV实战之形状识别的文章就介绍到这了,更多相关C++ OpenCV形状识别内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 使用Matlab绘制七夕咕呱小青蛙

    使用Matlab绘制七夕咕呱小青蛙

    七夕节到了还不快给你的朋友安排上这只咕呱小青蛙?本文将用Matlab绘制一个可爱的咕呱小青蛙,感兴趣的小伙伴可以动手尝试一下
    2022-08-08
  • Qt数据库应用之实现文件编码格式识别

    Qt数据库应用之实现文件编码格式识别

    在做数据导入导出的过程中,如果应用场景多了,相信各位都会遇到一个问题就是文件编码的问题。本文将用Qt实现文件编码格式识别,感兴趣的可以了解一下
    2022-06-06
  • C语言学习之关键字的示例详解

    C语言学习之关键字的示例详解

    关键字,这名字一听,就很关键。而有些关键字,你可能不是很了解,更别谈使用。所以,这篇文章将带你见识常见的关键字,一起领略它们的风采吧
    2022-10-10
  • C语言调用Python代码的方法

    C语言调用Python代码的方法

    这篇文章主要介绍了C语言调用Python代码的方法,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-07-07
  • C++初识类和对象

    C++初识类和对象

    类是创建对象的模板,一个类可以创建多个对象,每个对象都是类类型的一个变量;创建对象的过程也叫类的实例化。每个对象都是类的一个具体实例(Instance),拥有类的成员变量和成员函数
    2021-10-10
  • C语言杨氏矩阵查找算法实例讲解

    C语言杨氏矩阵查找算法实例讲解

    杨氏矩阵是一个数字矩阵,矩阵的每一行从左到右一次递增,矩阵从上到下递增,在这样的矩阵中查找一个数字是否存在。时间复杂度小于O(N),有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2022-09-09
  • C语言实现通讯录系统

    C语言实现通讯录系统

    这篇文章主要为大家详细介绍了C语言实现通讯录系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-02-02
  • C语言数据结构之二叉树的非递归后序遍历算法

    C语言数据结构之二叉树的非递归后序遍历算法

    这篇文章主要介绍了C语言数据结构之二叉树的非递归后序遍历算法的相关资料,希望通过本文能帮助到大家,让大家实现这样的功能,需要的朋友可以参考下
    2017-10-10
  • C++中构造函数与析构函数的详解及其作用介绍

    C++中构造函数与析构函数的详解及其作用介绍

    这篇文章主要介绍了C++中构造函数与析构函数的详解及其作用介绍,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-09-09
  • c调用python调试方法

    c调用python调试方法

    在本文里我们给大家分享了C中调用python调试的方法和相关知识点,需要的朋友们参考下。
    2019-02-02

最新评论