当前位置: 首页 > article >正文

OpenCV C++霍夫圆查找

OpenCV 中的霍夫圆检测基于 霍夫变换 (Hough Transform),它是一种从边缘图像中识别几何形状的算法。霍夫圆检测是专门用于检测图像中的圆形形状的。它通过将图像中的每个像素映射到可能的圆参数空间,来确定哪些像素符合圆形状。

1. 霍夫变换的原理

霍夫变换的基本思想是将图像空间中的点映射到一个参数空间,在这个参数空间中可以检测特定的几何形状(如直线、圆等)。对于圆形,霍夫变换的目标是找到符合圆方程的像素。

圆的方程

在二维平面上,圆可以由以下方程描述:

(x-a)^2+(y-b)^2=r^2

其中:

  • (a, b) 是圆心坐标;
  • r 是圆的半径;
  • (x, y) 是圆周上的一个点。

2. 霍夫圆检测的步骤

2.1 边缘检测
  • 在进行霍夫变换之前,首先对图像执行边缘检测,通常使用 Canny 边缘检测器。边缘检测的目的是找到图像中的边缘像素,因为这些像素更有可能属于圆周。
2.2 参数空间的投票
  • 每一个在边缘图像中的像素点 (x, y),它可能属于多个不同圆的边界,因此需要通过以下步骤将其投票映射到参数空间:

    • 将每个边缘像素映射为一组可能的圆心 (a, b),这些圆心位于距离该像素 r 的位置。通过改变半径 r ,该像素 (x, y) 将投票给不同半径下的多个圆心。
    • 对于每个可能的半径 r ,根据圆方程:

a=x-rcos\theta\\ b=y-rsin\theta

                其中 θ 是不同的角度值(从 0 到 360 度),遍历这些角度得到可能的圆心 (a, b)。

  • 每个像素点对不同半径 r 和不同圆心 (a, b) 进行投票,记录这些投票结果。

2.3 累加器
  • 在参数空间 (a, b, r) 中有一个累加器,用来统计哪些 (a, b, r) 的组合收到了最多的投票。最多投票的点表示在边缘图像中最可能的圆心和半径。
  • 累加器的最大值对应的是检测到的圆。
2.4 圆的确定
  • 最后,累加器中投票结果最多的那些 (a, b, r) 组合将被认为是图像中检测到的圆形。霍夫圆检测可以输出这些圆的圆心坐标 (a, b) 以及半径 r。

3. OpenCV 中霍夫圆检测的实现

OpenCV 中使用 HoughCircles() 函数来进行霍夫圆检测,其主要参数包括:

  • image:输入的边缘检测图像(通常是经过边缘检测或灰度化的图像)。
  • method:霍夫变换的检测方法,通常为 HOUGH_GRADIENT,这是经典的霍夫圆检测方法。
  • dp:累加器分辨率的倒数。 dp=1 表示累加器的分辨率与输入图像相同; dp=2 表示累加器分辨率是输入图像的一半。
  • minDist:检测到的圆之间的最小距离,防止检测到多个重叠的圆。
  • param1:用于边缘检测的 Canny 边缘检测的阈值。
  • param2:累加器的阈值,值越高检测到的圆越少,越准确。
  • minRadius  maxRadius:定义圆的最小和最大半径范围。
#include <opencv2/opencv.hpp> 
//#include "quickopencv.h"
#include <iostream>
#include <math.h>
#include <opencv2/imgproc.hpp>
#include <vector>

using namespace cv;
using namespace std;

int main(int argc, char** argv) {
	Mat img = imread("path_to_img.jpg");
	//QuickDemo qd;
	//qd.hough_circle(img);
	resize(img, img, Size(img.cols, img.rows), 0, 0, INTER_LINEAR);
	imshow("img", img);
	if (img.empty()) {
		cout << "请确认图像文件名称是否正确" << endl;
		return -1;
	}
	Mat gary;
	//将图像转换为灰度图
	cvtColor(img, gary, COLOR_BGR2GRAY);
	//高斯模糊
	//GaussianBlur(输入图像,输出图像,高斯核,X轴上的标准差,Y轴上的标准差);
	GaussianBlur(gary, gary, Size(9, 9), 2, 2);

	// 转换为二值图
	/*Mat binary;
	threshold(img, binary, 100, 255, cv::THRESH_BINARY);
	imshow("binary", binary);*/

	//检测圆形
	vector<Vec3f> circles;
	double dp = 1;		//累加器分辨率与图像分辨率的反比,如果dp=1,则累加器具有与输入图像相同的分辨率。如果dp=2,累加器的宽度和高度都是原来的一半。
	double minDist = 10;	//两个圆心的最小距离
	double param1 = 40;	//Canny边缘检测的较大阈值
	double param2 = 40;	//累加器阈值
	int min_radius = 1;	//圆形半径最小值
	int max_radius = 50;	//圆形半径最大值
	HoughCircles(gary, circles, HOUGH_GRADIENT, dp, minDist, param1, param2, min_radius, max_radius);


	//在图像中标记出圆形
	for (size_t i = 0; i < circles.size(); i++) {
		//读取圆心
		Point center(cvRound(circles[i][0]), cvRound(circles[i][1]));
		//读取半径
		int radius = cvRound(circles[i][2]);
		//绘制圆心
		circle(img, center, 3, Scalar(0, 255, 0), -1, 8, 0);
		// 设置圆心坐标的文本
		std::string centerText = "(" + std::to_string(center.x) + "," + std::to_string(center.y) + ")";
		// 计算文本框的大小
		cv::Size textSize = cv::getTextSize(centerText, cv::FONT_HERSHEY_SIMPLEX, 0.5, 1, nullptr);
		// 计算文本框的左下角位置,使其在圆心附近
		cv::Point textOrg((center.x - textSize.width / 2), (center.y + textSize.height / 2));
		// 在圆心处绘制文本
		cv::putText(img, centerText, textOrg, cv::FONT_HERSHEY_SIMPLEX, 0.5, cv::Scalar(255, 255, 255), 1);
		//绘制圆
		circle(img, center, radius, Scala

4. 霍夫圆检测的应用场景

霍夫圆检测广泛应用于多个场景,包括但不限于:

  • 车轮、硬币等圆形物体检测;
  • 医学图像中细胞、眼球、病变区域的圆形检测;
  • 机器视觉中的工业零件检测。


http://www.kler.cn/news/330235.html

相关文章:

  • Unity 编辑器多开
  • 前端框架Angular 详细学习要点
  • 性能测试的方式有哪些
  • 【评测大语言模型(LLM)的效果】
  • 实战OpenCV之边缘检测
  • CentOS 系统中的文件挂载 U 盘
  • ComfyUI自动提取lora触发词
  • 物联网网络中集中式与分布式SDN环境的比较分析
  • FPGA实现PCIE视频采集转HDMI输出,基于XDMA中断架构,提供3套工程源码和技术支持
  • JS中Object和Array的相互转换:深入全面讲解
  • STM32正交编码器的结构与工作原理
  • 数字化采购管理革新:全过程数字化采购管理平台的架构与实施
  • vue中子组件怎么修改父组件中的css样式的width值
  • stack、heap、.bss、.data、.text
  • Vue中对数组变化监听
  • python之with
  • Redis-常见数据类型(修改ing)
  • 《PMI-PBA认证与商业分析实战精析》第5章 需求启发与分析
  • Linux:深入理解冯诺依曼结构与操作系统
  • 企业数据挖掘工具选择指南:如何找到最适合的工具