OpenCv-C++-小案例实战-对象提取

现在比如说我有这么一张图:
在这里插入图片描述
我想要从中提取出"圆"的部分,那该怎么做呢?用霍夫圆变换?No,万一它不是一个标准的圆呢?所以,这里采用轮廓发现的方法来查找"圆"。

解决思路:
1、二值化;
2、形态学操作(开运算+闭运算);
3、轮廓点查找;
4、坐标查找。

首先将图像进行二值化,轮廓点查找需要二值化图像。先使用闭操作去除图片最上面那一大块中的白色小点,这可以把它叫做联通。再使用开操作去除其余的黑色小点。这样操作之后我们就可以进行轮廓点查找了,算出该区域的面积,如果面积<100(这个值可以根据情况取),则不会绘制轮廓。之后找出最小外接矩形,把它的长宽进行相除,如果该比例在1上下左右进行浮动,则被认为是该“圆”。

代码:

#include<opencv2/opencv.hpp>
#include<iostream>
#include<math.h>

using  namespace cv;
using  namespace std;

Mat src, binary, close_dst,open_dst;
int main(int argc, char** argv)
{
	src = imread("D:/test/object.png", 0);
	if (src.empty())
	{
		cout << "图片未找到!!!" << endl;
		return -1;
	}
	imshow("input image", src);

	//二值化操作
	threshold(src, binary, 0, 255, THRESH_BINARY | THRESH_OTSU);
	bitwise_not(binary, binary, Mat());
	imshow("binary image",binary);

	/*---------------------形态学操作-------------------------*/
	Mat kernel = getStructuringElement(MORPH_RECT, Size(3, 3), Point(-1, -1));
	//闭操作,把中间的细小点联通起来
	morphologyEx(binary, close_dst, MORPH_CLOSE, kernel, Point(-1, -1));
	imshow("close image", close_dst);
	//开操作,去除多余的点
	kernel = getStructuringElement(MORPH_RECT, Size(7, 7), Point(-1, -1));
	morphologyEx(binary, open_dst, MORPH_OPEN, kernel, Point(-1, -1));
	imshow("open image", open_dst);
	/*---------------------获取对象轮廓-------------------------*/
	Mat dst = Mat::zeros(src.size(), CV_8UC3);
	vector <vector<Point>> contours;
	vector<Vec4i> hierachy;
	findContours(open_dst, contours, hierachy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));
	Point cc;
	Mat result_img = src.clone();
	cvtColor(result_img, result_img, CV_GRAY2BGR);

	//RNG rng((int)time(0));
	for (size_t t = 0; t < contours.size(); t++)
	{

		double area = contourArea(contours[t]);//获取每个轮廓围成的面积
		if (area < 100)  //如果轮廓围成的面积小于100,则过滤
			continue;
		//横纵比过滤
		Rect rect = boundingRect(contours[t]);//用最小的外接矩形把对象给包起来
		//因为检测提取的对象是圆,所以横纵比可以设定为在0.9-1.2之间
		float h_w = (float)rect.height / rect.width;
		if (h_w > 0.9&&h_w < 1.2)
		{
		drawContours(dst, contours, static_cast<int>(t), Scalar(0, 0, 255), 2, 8, Mat(), 0, Point(0, 0));
		double length = arcLength(contours[t], true);//周长
		printf("面积:%f\n",area);
		printf("周长:%f\n", length);

		/*----------------获取圆点坐标----------------------*/
		
		int x = rect.x + rect.width / 2;//rect.x是矩形左上角的横坐标
		int y = rect.y + rect.height / 2;//rect.y是矩形左上角的纵坐标
		cc = Point(x, y);
		//Scalar color = Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255));
		circle(result_img, cc, 2, Scalar(0,0,255), 2, 8, 0);

		}
	}
	
	imshow("contours image", dst);
	imshow("result image", result_img);
	waitKey(0);
	return 0;


}

运行结果:
输入图片:
在这里插入图片描述

进行二值化后的结果:
在这里插入图片描述

闭操作之后的结果:
在这里插入图片描述
可以看到,黑色小点被去除了。

开操作的结果:
在这里插入图片描述
这里比较好,只进行了开操作就把其余的小块区域给去除了。

轮廓发现:
在这里插入图片描述

最后在原图上显示圆点坐标:
在这里插入图片描述

在这里插入图片描述

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
课程目的:OpenCV是应用非常广泛的开源视觉处理库,在图像处理、计算机视觉和自动驾驶中有着非常重要的作用。课程设计特色:(课程当前为第一期)1、C++与Python双语教学Python语言是在计算机视觉中应用最多的一种语言,在工作中,深度学习模型的训练基本上都是使用Python语言编写的训练代码。OpenCV在这个过程中用于图像的预处理(例如图像读取、数据增强)和后处理,还可以用于显示处理的结果,功能强大,使用方便。但是在功能的部署的时候,不管是部署在服务端还是PC端,开发语言基本上用的是C++,所以如何有效的使用OpenCV进行模型或者功能的部署尤为重要。C++语言应用的好坏,在面试中可以看出一个面试者的工程实践能力的强弱,两种语言的开发掌握好了可以使工作如虎添翼。2、全模块讲解我出版了一本图书《学习OpenCV4:基于Python的算法实战》,虽然这本书是写的基于Python的算法实战,但是实际上这本书有详细的介绍算法的C++口,还有一些C++方向的案例,是以Python为主。图书出版的时候就想双语写作,只是限于篇幅没有成行。本课程不仅采用双语教学,更是对C++的每个模块都做讲解,我们知道,很多的书其实只讲imgproc,如果你翻开一本书图像的形态学运算和图像滤波都是作为独立章节讲解的,那么这本书基本上就可以确定是只是讲解了imgproc模块,但是其他的模块在工作中也有很重要的作用。例如:core模块定义了C++的基本数据结构和基本运算(如四则运算);highgui模块是可视化与交互的模块;feature2d是特征与特征匹配相关算法所在的模块;ml是机器学习相关的模块;dnn是深度学习相关的模块,可以使用OpenCV进行深度学习模型的部署。这些是很多的书和课程都不会讲的。3、讲解细致本课程会从环境搭建开始讲解,环境搭建尤为重要。从我多年的授课经验总结来看,如果只是给了代码,很多的入门用户环境问题处理不好的话,后面的学习很难进行下去,甚至会丧失学习的信心。4、会讲解C++和Python的开发语法问题是入门用户的一大难关,特别是C++语言。大学只是教授了C语言相关的内容,C++很多同学只懂一皮毛,所以写代码步履维艰,我们在讲解代码的过程中会顺带讲解C++和Python的内容。我们还会讲解编译相关知识,还有库的装载与链,这些是学校里不会教的,目前也几乎没有课程讲解。5、讲师经验丰富我讲解过C++OpenCV的多个课程,广受学员好评。我出版过两本图书《深度学习计算机视觉实战》和《学习OpenCV4》,两本书都是细致入微的讲解,主要针对的就是初学者,所以能够很好的处理课程的难易程度。6、讲义准备充分讲义准备的充分细致,标识清楚明确,重和疑难突出。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值