基于OpenCV的霍夫直线检测和霍夫圆形检测

一:霍夫直线检测

1. 霍夫变换简介 

霍夫变换是一种特征检测(feature extraction),被广泛应用在图像分析(image analysis)、电脑视觉(computer vision)以及数位影像处理(digital image processing)。霍夫变换是用来辨别找出物件中的特征,例如:线条。他的算法流程大致如下,给定一个物件、要辨别的形状的种类,算法会在参数空间(parameter space)中执行投票来决定物体的形状,而这是由累加空间(accumulator space)里的局部最大值(local maximum)来决定。

2.霍夫线变换 

OpenCV中的霍夫线变换有如下三种: 

<1> 标准霍夫变换(StandardHough Transform,SHT),由HoughLines函数调用。

<2> 多尺度霍夫变换(Multi-ScaleHough Transform,MSHT),由HoughLines函数调用。

<3> 累计概率霍夫变换(ProgressiveProbabilistic Hough Transform,PPHT),由HoughLinesP函数调用。
 

cvHonghLines2() 函数:

说明:

  此函数是opencv图像变换函数中的一个,主要用来访问霍夫变换的两个算法———标准霍夫变换(SHT)和累计概率霍夫变换(PPHT)。

函数原型:

   CvSeq* cvHonghLines2(
   CvArr* image,
   void* line_storage,
   int mehtod,
   double rho,
   double theta,
   int threshold,
   double param1 =0,
   double param2 =0
   );

参数说明:

  image
  输入 8-比特、单通道 (二值) 图像,当用CV_HOUGH_PROBABILISTIC方法检测的时候其内容会被函数改变。
  line_storage
  检测到的线段存储仓. 可以是内存存储仓 (此种情况下,一个线段序列在存储仓中被创建,并且由函数返回),或者是包含线段参数的特殊类型(见下面)的具有单行/单列的矩阵(cvMat*)。矩阵头为函数所修改,使得它的 cols/rows 将包含一组检测到的线段。如果 line_storage 是矩阵,而实际线段的数目超过矩阵尺寸,那么最大可能数目的线段被返回(线段没有按照长度、可信度或其它指标排序).
  method
  Hough 变换变量,是下面变量的其中之一:
  CV_HOUGH_STANDARD - 传统或标准 Hough 变换. 每一个线段由两个浮点数 (ρ, θ) 表示,其中 ρ 是直线与原点 (0,0) 之间的距离,θ 线段与 x-轴之间的夹角。因此,矩阵类型必须是 CV_32FC2 type.
  CV_HOUGH_PROBABILISTIC - 概率 Hough 变换(如果图像包含一些长的线性分割,则效率更高). 它返回线段分割而不是整个线段。每个分割用起点和终点来表示,所以矩阵(或创建的序列)类型是 CV_32SC4.
  CV_HOUGH_MULTI_SCALE - 传统 Hough 变换的多尺度变种。线段的编码方式与 CV_HOUGH_STANDARD 的一致。
  rho 
  与象素相关单位的距离精度
  theta 
  弧度测量的角度精度
  threshold
  阈值参数。如果相应的累计值大于 threshold, 则函数返回的这个线段.
  param1
  第一个方法相关的参数:
  对传统 Hough 变换,不使用(0).
  对概率 Hough 变换,它是最小线段长度.
  对多尺度 Hough 变换,它是距离精度 rho 的分母 (大致的距离精度是 rho 而精确的应该是 rho / param1 ).
  param2
  第二个方法相关参数:
  对传统 Hough 变换,不使用 (0).
  对概率 Hough 变换,这个参数表示在同一条直线上进行碎线段连接的最大间隔值(gap), 即当同一条直线上的两条碎线段之间的间隔小于param2时,将其合二为一。
  对多尺度 Hough 变换,它是角度精度 theta 的分母 (大致的角度精度是 theta 而精确的角度应该是 theta / param2).

实例:

	IplImage* src;
	IplImage* dst2;
	double distance;

	src = cvLoadImage(szDealPic);
	if (src != NULL)
	{
		IplImage* dst = cvCreateImage(cvGetSize(src), 8, 1);
		IplImage* color_dst = cvCreateImage(cvGetSize(src), 8, 3);
		CvMemStorage* storage = cvCreateMemStorage(0);//存储检测到线段,当然可以是N*1的矩阵数列,如果实际的直线数量多余N,那么最大可能数目的线段被返回
		CvSeq* lines = 0;
		int i;
		IplImage* src1 = cvCreateImage(cvSize(src->width, src->height), IPL_DEPTH_8U, 1);

		cvCvtColor(src, src1, CV_BGR2GRAY); //把src转换成灰度图像保存在src1中,注意进行边缘检测一定要换成灰度图
		cvCanny(src1, dst, 50, 200, 3);//参数50,200的灰度变换

		cvCvtColor(dst, color_dst, CV_GRAY2BGR);

		//cvThreshold(dst, dst2, 100, 255, CV_THRESH_OTSU);  //全局自适应 CV_THRESH_OTSU  // 原:CV_THRESH_BINARY

		lines = cvHoughLines2(dst, storage, CV_HOUGH_PROBABILISTIC, 1, CV_PI / 180, 80, 30, 10);
		for (i = 0; i < lines->total; i++)
		{
			CvPoint* line = (CvPoint*)cvGetSeqElem(lines, i);
			cvLine(color_dst, line[0], line[1], CV_RGB(255, 255, 0), 3, 8);
			double k = (double)(line[1].y - line[0].y) / (double)(line[1].x - line[0].x);
			double d = line[0].y - line[0].x*k;
			if (i == 0)
			{
				d1 = d;
			}
			else
			{
				d2 = d;
			}
			LOG_FOR_RD("This is the %d line, this line is y = %.2f x + %.2f \n", i + 1, k, d);
		}

		distance = abs(d1 - d2);
	}

 

 

 二:霍夫圆检测

cv2中进行霍夫圆环检测的函数:

cv2.HoughCircles(image, method, dp, minDist, circles=None, param1=None, param2=None, minRadius=None, maxRadius=None)

其中:

image:8位,单通道图像。如果使用彩色图像,需要先转换为灰度图像。

method:定义检测图像中圆的方法。目前唯一实现的方法是cv2.HOUGH_GRADIENT。

dp:累加器分辨率与图像分辨率的反比。dp获取越大,累加器数组越小。

minDist:检测到的圆的中心,(x,y)坐标之间的最小距离。如果minDist太小,则可能导致检测到多个相邻的圆。如果minDist太大,则可能导致很多圆检测不到。

param1:用于处理边缘检测的梯度值方法。

param2:cv2.HOUGH_GRADIENT方法的累加器阈值。阈值越小,检测到的圈子越多。

minRadius:半径的最小大小(以像素为单位)。

maxRadius:半径的最大大小(以像素为单位)。
 

代码实例:

	::GetModuleFileNameA(NULL, szModuleName, 256);
	char *p = strrchr(szModuleName, '\\');
	if (p)
	{
		strncpy_s(szWorkDir, szModuleName, p - szModuleName + 1);//including the '\' character
	}

	sprintf(ImagePath, "%slena.bmp", szWorkDir);
	//about cycles 
	// _CRT_SECURE_NO_WARNINGS

	Mat midImage;
	Mat srcImage = imread(ImagePath);

	//imshow("【原始图】", srcImage);
	cvtColor(srcImage, midImage, CV_BGR2GRAY);//转化边缘检测后的图为灰度图
	GaussianBlur(midImage, midImage, Size(9, 9), 2, 2);

	vector<Vec3f> circles;
	HoughCircles(midImage, circles, CV_HOUGH_GRADIENT, 1, midImage.rows / 20, 100, 100, 0, 0);/ exe has triggered a breakpoint
	//HoughCircles(midImage, circles, CV_HOUGH_GRADIENT, 2, midImage.rows / 4, 200, 100);
	//return 0;
	for (size_t i = 0; i < circles.size(); i++)
	{
		Point center(cvRound(circles[i][0]), cvRound(circles[i][1]));
		//计算半径
		radius = cvRound(circles[i][2]);		
		//绘制圆心		
		circle(srcImage, center, 3, Scalar(0, 255, 0), -1, 8, 0);		
		//绘制圆轮廓		
		circle(srcImage, center, radius, Scalar(155, 50, 255), 3, 8, 0);

		//printf("=%d", radius);
	}

参考:https://blog.csdn.net/happy_stars_2016/article/details/52691255

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
import cv2 as cv import numpy as np def hough_circle(image): #因为霍夫检测对噪声很明显,所以需要先滤波一下。 dst =cv.pyrMeanShiftFiltering(image,10,100) cimage=cv.cvtColor(dst,cv.COLOR_BGR2GRAY) circles = cv.HoughCircles(cimage,cv.HOUGH_GRADIENT,1,40,param1=40,param2=29,minRadius=30,maxRadius=0) #把circles包含的圆心和半径的值变为整数 circles = np.uint16(np.around(circles)) for i in circles[0]: cv.circle(image,(i[0],i[1]),i[2],(0,255,0),3) cv.imshow("circle",image) src = cv.imread("E:/opencv/picture/coins.jpg") cv.imshow("inital_window",src) hough_circle(src) cv.waitKey(0) cv.destroyAllWindows() 霍夫圆变换的基本思路是认为图像上每一个非零像素点都有可能是一个潜在的圆上的一点, 跟霍夫线变换一样,也是通过投票,生成累积坐标平面,设置一个累积权重来定位圆。 在笛卡尔坐标系中圆的方程为: 其中(a,b)是圆心,r是半径,也可以表述为: 即 在笛卡尔的xy坐标系中经过某一点的所有圆映射到abr坐标系中就是一条三维的曲线: 经过xy坐标系中所有的非零像素点的所有圆就构成了abr坐标系中很多条三维的曲线。 在xy坐标系中同一个圆上的所有点的圆方程是一样的,它们映射到abr坐标系中的是同一个点,所以在abr坐标系中该点就应该有圆的总像素N0个曲线相交。 通过判断abr中每一点的相交(累积)数量,大于一定阈值的点就认为是圆。 以上是标准霍夫圆变换实现算法。 问题是它的累加到一个三维的空间,意味着比霍夫线变换需要更多的计算消耗。 Opencv霍夫圆变换对标准霍夫圆变换做了运算上的优化。 它采用的是“霍夫梯度法”。它的检测思路是去遍历累加所有非零点对应的圆心,对圆心进行考量。 如何定位圆心呢?圆心一定是在圆上的每个点的模向量上,即在垂直于该点并且经过该点的切线的垂直线上,这些圆上的模向量的交点就是圆心。 霍夫梯度法就是要去查找这些圆心,根据该“圆心”上模向量相交数量的多少,根据阈值进行最终的判断。 bilibili: 注意: 1.OpenCV霍夫圆变换函数原型为:HoughCircles(image, method, dp, minDist[, circles[, param1[, param2[, minRadius[, maxRadius]]]]]) -> circles image参数表示8位单通道灰度输入图像矩阵。 method参数表示圆检测方法,目前唯一实现的方法是HOUGH_GRADIENT。 dp参数表示累加器与原始图像相比的分辨率的反比参数。例如,如果dp = 1,则累加器具有与输入图像相同的分辨率。如果dp=2,累加器分辨率是元素图像的一半,宽度和高度也缩减为原来的一半。 minDist参数表示检测到的两个圆心之间的最小距离。如果参数太小,除了真实的一个圆圈之外,可能错误地检测到多个相邻的圆圈。如果太大,可能会遗漏一些圆圈。 circles参数表示检测到的圆的输出向量,向量内第一个元素是圆的横坐标,第二个是纵坐标,第三个是半径大小。 param1参数表示Canny边缘检测的高阈值,低阈值会被自动置为高阈值的一半。 param2参数表示圆心检测的累加阈值,参数值越小,可以检测越多的假圆圈,但返回的是与较大累加器值对应的圆圈。 minRadius参数表示检测到的圆的最小半径。 maxRadius参数表示检测到的圆的最大半径。 2.OpenCV画圆的circle函数原型:circle(img, center, radius, color[, thickness[, lineType[, shift]]]) -> img img参数表示源图像。 center参数表示圆心坐标。 radius参数表示圆的半径。 color参数表示设定圆的颜色。 thickness参数:如果是正数,表示圆轮廓的粗细程度。如果是负数,表示要绘制实心圆。 lineType参数表示圆线条的类型。 shift参数表示圆心坐标和半径值中的小数位数。
霍夫直线检测Hough Line Transform)是一种常用的图像处理技术,用于检测图像中的直线。OpenCV库中包含了对霍夫直线检测的支持。 在OpenCV中,使用`HoughLines`函数进行霍夫直线检测。函数的基本使用方法如下: ```python lines = cv2.HoughLines(image, rho, theta, threshold) ``` 参数说明: - `image`:输入的二值化图像,通常为边缘检测后的图像。 - `rho`:表示ρ(rho)参数的精确度,一般取1。 - `theta`:表示θ(theta)参数(角度)的精确度,一般取π/180。 - `threshold`:用于筛选直线的阈值,只有当累加器中的值大于阈值时,才被认为是一条直线。 函数返回的是一个由直线参数组成的数组,每个直线参数是一个(rho, theta)对,表示检测到的一条直线。可以根据需要对返回的直线进行后续处理或绘制。 另外,OpenCV还提供了`HoughLinesP`函数,它可以直接返回直线在图像上的起点和终点坐标,方便进行直线的绘制。 ```python lines = cv2.HoughLinesP(image, rho, theta, threshold, minLineLength, maxLineGap) ``` 除了上述参数外,`HoughLinesP`还包含两个额外的参数: - `minLineLength`:表示线段的最小长度,小于该长度的线段会被排除。 - `maxLineGap`:表示两条线段之间的最大间隔,超过该间隔的线段会被视为不同的线段。 希望这些信息能帮助到你进行霍夫直线检测!如有更多问题,请继续提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值