OpenCV单目视觉定位(测量)系统(新增 含代码)

OpenCV单目视觉定位(测量)系统

The System of Vision Location with Signal Camera



Abstract:This passage mainly describes how to locate with signalcamera,which bases on OpenCV library.

Key words: OpenCV; Locate;Signalcamera

摘  要本文主要描述的是利用开源计算机视觉库OpenCV实现单目视觉定位系统。

关键词: OpenCV;单位视觉定位

1     总体设计方案

单目视觉定位系统是通过单个分辨率为640*480,20万的USB手动调焦摄像头获取图片并传到计算机利用OpenCV进行预处理、识别、定位、测量等图像处理算法,进而得出目标物体相对摄像头的二维坐标距离。

2     预处理

对摄像头获取的图片进行均值滤波处理,目的是抑制电气元件或者环境因素引起的噪声。滤波也叫平滑处理,滤波的目的是抑制噪声,平滑滤波属于低频增强空间滤波技术,因为图像的能量大部分集中在幅度谱的低中频,噪声一般存在于高频段。

在本系统中主要采用均值滤波算法,也叫邻域平均算法。领域平均算法的基本思想是对含噪声图像的每一个像素点f(x,y)取领域S,用领域S中所包含的像素灰度平均值来代替该点的灰度值。

领域平均法的优点是处理方法简计算速度快,缺点是在降低噪声的同时是图像产生一定程度的模糊,领域半径越大,去噪效果越好,但图像越模糊。

因此,本系统采用3 X 3模板h,其转置矩阵为:1/9 [1,1,1 ; 1,1,1 ; 1,1,1],设f(x,y)为原图像的每一个像素值,g(x,y)为输出图像对应的每一个像素值,则均值滤波核心算法为原图像的每一个像素(除图像边缘像素)和3 X 3模板h进行卷积运算得出输出图像。

 图一 灰度图                 


图二 均值滤波                           


3     识别

对于目标物体的识别,本系统是采用Canny边缘检测找出所有可能的边缘信息,然后对边缘信息进行提取和分析,进而识别出目标物体。与该目标物体的识别还有一种方法是采用阈值算法分割出特定灰度值范围内的物体,但该算法受光照影响较大,而边缘检测受光照影响较少,因此Canny采用边缘检测算法。

3.1目标物体

对目标物体的设计,我采用了外面六边形,里面五边形,六边形与五边形之间黑色填充,其余白色填充,六边形与五边形中心基本上一致。如下图所示


 图三打印的目标物体

 

 3.2 Canny边缘检测

边缘检测的算法只要是基于图像增强的一阶和二阶导数。Canny边缘检测算子是JohnF.Canny于1986年开发出来的一个多级边缘检测算法。Canny的目标是找到一个最优的边缘检测算法。其具有的优点是:

l  低错误性:标识出尽可能多的实际边缘,同时尽可能地减少噪声产生的误报。

l  高定位性:标识出的边缘要与图像中的实际边缘尽可能接近

l  最小效应:图像中的边缘只能标识一次,并且可能存在的图像噪声不应标识为边缘

其步骤是,首先使用高斯平滑滤波器卷积降噪,计算梯度幅值和方向,然后进行非极大值抑制,排除非边缘像素,仅仅保留了一些细线条,最后滞后阈值。

对上面经过均值滤波的灰度图进行Canny边缘检测。


 图四 边缘检测图

 

3.3 轮廓分析

对上面的Canny边缘检测得出的信息运用OpenCV提供的findContours函数边缘信息转化为轮廓信息,最后再对轮廓进行分析处理,找出目标物体。

 

图五 轮廓图

轮廓分析算法:遍历所有轮廓,排除凹轮廓和轮廓面积小于100像素的轮廓,找出轮廓为逼近六边形和逼近五边形的所有轮廓,找出这些逼近六边形和逼近五边形轮廓的最小外接矩形,并各提取最小外接矩形的中心(即逼近六边形和逼近五边形轮廓中心),若逼近六边形轮廓的最小外接矩形和逼近五边形轮廓的最小外接矩形中心基本在同一点(考虑畸变原因,中心点不一定一致),则为目标物体。

3       定位

本系统是采用640*480分辨率,20万像素的手动调焦的USB摄像头进行实验的,该摄像头广角较低,畸变不厉害。

定位功能得出的坐标是相对于自定义的图像坐标原点,在该系统中,定义的原点就是图像中心(320,240)单位为像素。因此,在摄像头水平放置的情况下,自定义图像原点即为二维空间上的摄像头正下方。

定位的基本思想是利用已知目标物体的长度和求出的目标物体的像素长度,得出像素的尺寸,再利用像素尺寸和目标物体在自定义原点的X,Y方向的像素距离,进而求出目标物体相对水平放置的摄像头的X,Y方向实际距离。

实验结果,本实验是摄像头距目标物体正上方大约50cm上进行测量的,摄像头为水平放置。经过多次测量,测出距离与实际距离误差基本上维持在1cm到2cm之间。如下图所示:(红色点为自定义的图象原点,左上方的Target_X,Target_Y分别为系统测出的目标距原点(摄像头正下方)的X和Y方向的测出距离。图像上的白色坐标系为实际理论距离。

 

图六 实验结果图

 


                                                                               图、实验情况

1     结束语

该测量系统只在短范围内测量,实验还存在误差。因为该摄像头本身畸变不大,没有进行标定,矫正等一系列处理,实验数据还能接受,但对于广角大、畸变大的摄像头,本系统根本无法适用。还有的是,本系统还没有进行相机姿态估计,因此,对于摄像头与水平面有一定角度时,误差会大大增加。

在接下来的时间里,作者会尽量完善该系统,争取能用于无人机以及机器人的单目视觉定位系统。

本文难免有错漏,欢迎大家指出,本人联系方式:821992904@qq.com


核心代码:

//定位函数
void Location(){

	//均值滤波
	blur(gray, gray_blur, Size(3, 3));

	//边缘检测提取边缘信息
	Canny(gray_blur, dstThreshold, 150, 450);
	imshow("canny边缘检测", dstThreshold);
	
	//对边缘图像提取轮廓信息
    vector<vector<Point> >contours;
	findContours(dstThreshold, contours, CV_RETR_LIST, CV_CHAIN_APPROX_SIMPLE);
	
	//画出轮廓
	drawContours(contours_image, contours, -1, Scalar(0, 0, 255));
	imshow("contours", contours_image);

	//画出定义的原点
	circle(src, Point2f(oriX, oriY), 2, Scalar(0, 0, 255), 3);

	//定义分别逼近六边形和五边形的轮廓
	vector< vector<Point> > Contour1_Ok, Contour2_Ok;

	//轮廓分析
	vector<Point> approx;
	for (int i = 0; i < contours.size(); i++){
		approxPolyDP(Mat(contours[i]), approx, cv::arcLength(cv::Mat(contours[i]), true)*0.04, true);

		//去除 小轮廓,只提取凸轮廓
		if (std::fabs(cv::contourArea(contours[i])) < 600 || !cv::isContourConvex(approx))
			continue;

	    //保存逼近六边形的轮廓 到 Contour1_Ok
		 if (approx.size() == 6){
			Contour1_Ok.push_back(contours[i]);
		}
	    //保存逼近五边形的轮廓 到 Contour2_Ok
		 else if (approx.size() == 5){
			 Contour2_Ok.push_back(contours[i]);
		}

	}

	//对所有符合要求的六边形,五边形轮廓进行分析
	//识别出自定义的物体的关键是:
	//1.六边形和五边形轮廓的最小外接矩形的中心基本在同一点
	//2.六边形轮廓的最小外接矩形的任一边长大于五边形轮廓的最小外接矩形的任一边长
	for (int i = 0; i < Contour1_Ok.size(); i++){
		for (int j = 0; j < Contour2_Ok.size(); j++){
			RotatedRect minRect1 = minAreaRect(Mat(Contour1_Ok[i]));  //六边形轮廓的最小外接矩形
			RotatedRect minRect2 = minAreaRect(Mat(Contour2_Ok[j]));  //五边形轮廓的最小外界矩形
			//找出符合要求的轮廓的最小外接矩形
			if ( fabs(minRect1.center.x - minRect2.center.x) < 30 && fabs(minRect1.center.y - minRect2.center.y)<30 && minRect1.size.width > minRect2.size.width){
				Point2f vtx[4];
				minRect1.points(vtx);
				
				//画出找到的物体的最小外接矩形
				for (int j = 0; j < 4; j++)
					line(src, vtx[j], vtx[(j + 1) % 4], Scalar(0, 0, 255), 2, LINE_AA);

				//画出目标物中心到图像原点的直线
				line(src, minRect1.center, Point2f(oriX, oriY), Scalar(0, 255, 0), 1, LINE_AA);

				//目标物距图像原点的X,Y方向的像素距离
				targetImage_X = minRect1.center.x - oriX;
				targetImage_Y = oriY - minRect1.center.y;

				line(src, minRect1.center, Point2f(minRect1.center.x, oriY), Scalar(255, 0, 0), 1, LINE_AA);
				line(src, Point2f(oriX, oriY), Point2f(minRect1.center.x, oriY), Scalar(255, 0, 0), 1, LINE_AA);

                Point2f pointX((oriX + minRect1.center.x) / 2, oriY);
				Point2f pointY(minRect1.center.x, (oriY + minRect1.center.y) / 2);  
		
		        //找出最大边
				float a = minRect1.size.height, b = minRect1.size.width;
				if (a < b) a = b;
				
				mm_per_pixel = targetLength / a;               //计算像素尺寸 = 目标物的实际长度(cm)/ 目标物在图像上的像素长度(pixels)
				targetActualX = mm_per_pixel *targetImage_X;   //计算实际距离X(cm)
				targetActualY = mm_per_pixel *targetImage_Y;   //计算实际距离Y(cm)

				//打印信息在图片上
				String text1 = "X:"+format("%f", targetImage_X);
				String text2 = "Y:"+format("%f", targetImage_Y);
				putText(src, text1, pointX, FONT_HERSHEY_SIMPLEX, 0.4, Scalar(0, 0, 255), 1, 8);
				putText(src, text2, pointY, FONT_HERSHEY_SIMPLEX, 0.4, Scalar(0, 0, 255), 1, 8);

				String text3 = "Target_X:"+format("%f", targetActualX);
				String text4 = "Target_Y:"+format("%f", targetActualY);
				putText(src, text3, Point(10,30), FONT_HERSHEY_SIMPLEX, 0.7, Scalar(0, 0, 255), 1, 8);
				putText(src, text4, Point(10,60), FONT_HERSHEY_SIMPLEX, 0.7, Scalar(0, 0, 255), 1, 8);

				
			}
			break;
		}       
        break;
	}

	imshow("SRC", src);

}

完整代码

http://download.csdn.net/download/chenmohousuiyue/9947409
点击打开链接



  • 30
    点赞
  • 449
    收藏
    觉得还不错? 一键收藏
  • 14
    评论
单目视觉定位是指利用单个摄像头获取的图像信息,通过计算机视觉算法来实现对目标在三维空间中的位置和姿态的估计。 在使用OpenCV进行单目视觉定位时,一般采用以下几个步骤: 1. 相机标定:首先需要对使用的相机进行标定,通过拍摄一系列已知空间位置的标定板图像,使用相机标定算法计算相机的内参(焦距、主点)和外参(旋转矩阵、平移向量),以便进行后续的定位计算。 2. 特征提取:从图像中提取出能够用于定位的特征点,例如角点、边缘等。OpenCV提供了多种特征提取算法,如SIFT、SURF、ORB等。 3. 特征匹配:将实时获取的图像特征点与已知的目标特征点进行匹配,以找到最优的匹配点对。多种特征匹配算法可用,如FLANN、汉明距离等。 4. 位姿估计:根据特征点的匹配结果,使用PnP算法估计相机的姿态(旋转矩阵、平移向量),并通过矩阵运算将其转换为目标在世界坐标系下的位置和姿态。 5. 定位反馈:根据目标在世界坐标系下的位置和姿态,可以进行相应的控制与反馈,例如机器人的导航、姿态调整等。 需要注意的是,单目视觉定位在实际应用中可能会面临光照变化、遮挡等问题,导致特征提取和匹配的准确性下降。解决这些问题可以使用多视角信息融合、滤波等技术。 总而言之,OpenCV作为一个开源的计算机视觉库,提供了丰富的功能和算法,可以方便地实现单目视觉定位。通过相机标定、特征提取和匹配、位姿估计等步骤,可以准确地估计目标在世界坐标系下的位置和姿态,为实际应用提供了强大的支持。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值