霍夫变换(直线检测、圆检测)

可以看出如果笛卡尔坐标系的点共线,这些点在霍夫空间对应的直线交于一点:这也是必然,共线只有一种取值可能。

如果不止一条直线呢?再看看多个点的情况(有两条直线):

 

 

 

 

 

 

 

在opencv中步骤解读:

具体步骤:

1. 彩色图像->灰度图

2. 去噪(高斯核)

3. 边缘提取(梯度算子、拉普拉斯算子、canny、sobel) 

4. 二值化(判断此处是否为边缘点,就看灰度值==255)

5. 映射到霍夫空间(准备两个容器,一个用来展示hough-space概况,一个数组hough-space用来储存voting的值,因为投票过程往往有某个极大值超过阈值,多达几千,不能直接用灰度图来记录投票信息)

6. 取局部极大值,设定阈值,过滤干扰直线

7. 绘制直线、标定角点

函数HoughLines的实现代码

#include "opencv2/core/core.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/opencv.hpp"
#include <iostream>
using namespace cv;
using namespace std;
using std::cout;
int main()
{

	Mat g_srcImage, midImage, dstImage;
	namedWindow("[原始图]");
	g_srcImage = imread("E:\\VS2015Opencv\\vs2015\\project\\picture\\01.jpg");
	if (!g_srcImage.data) { cout << "error read image" << endl; return 0; }

	imshow("[原始图]", g_srcImage);

	Canny(g_srcImage, midImage, 50, 200, 3);
	cvtColor(midImage, dstImage, CV_GRAY2BGR);

	vector<Vec2f> lines;
	HoughLines(midImage, lines, 0.5, CV_PI / 18, 150, 0, 0);

	for (size_t i = 0; i < lines.size(); i++)
	{
		float rho = lines[i][0], theta = lines[i][1];
		Point pt1, pt2;
		double a = cos(theta), b = sin(theta);
		double x0 = a*rho, y0 = b*rho;
		pt1.x = cvRound(x0 + 2000 * (-b));  //把浮点数转化成整数
		pt1.y = cvRound(y0 + 2000 * (a));
		pt2.x = cvRound(x0 - 2000 * (-b));
		pt2.y = cvRound(y0 - 2000 * (a));
		line(dstImage, pt1, pt2, Scalar(128, 128, 0), 1, CV_AA);
	}
	imshow("边缘检测后的图", midImage);
	imshow("标准霍夫线变换效果图", dstImage);

	waitKey(0);
	return 0;
}

  

 

 函数HoughLinesP的代码

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

using namespace cv;
using namespace std;


int main(int argc, char** argv)
{
    Mat src, src_gray, dst;
    src = imread("E:\\VS2015Opencv\\vs2015\\project\\picture\\01.jpg");

    char INPUT_TITLE[] = "input image";

    imshow(INPUT_TITLE, src);

    Canny(src, src_gray, 150, 200);
    cvtColor(src_gray, dst, CV_GRAY2BGR);
    imshow("edge image", src_gray);
    imshow("gray", dst);

    //方法1(标准霍夫变换)
    //vector<Vec2f> lines;
    //HoughLines(src_gray, lines, 1, CV_PI / 180, 150, 0, 0);
    //for (size_t i = 0; i < lines.size(); i++) {
    //    float rho = lines[i][0]; // 极坐标中的r长度
    //    float theta = lines[i][1]; // 极坐标中的角度
    //    Point pt1, pt2;
    //    double a = cos(theta), b = sin(theta);
    //    double x0 = a * rho, y0 = b * rho;
    //    // 转换为平面坐标的四个点
    //    pt1.x = cvRound(x0 + 1000 * (-b));//对一个double型的数进行四舍五入,并返回一个整型数!
    //    pt1.y = cvRound(y0 + 1000 * (a));
    //    pt2.x = cvRound(x0 - 1000 * (-b));
    //    pt2.y = cvRound(y0 - 1000 * (a));
    //    line(dst, pt1, pt2, Scalar(0, 0, 255), 1, CV_AA);
    //}


    //第二种方法(概率霍夫变换)
    vector<Vec4f> plines;
    HoughLinesP(src_gray, plines, 1, CV_PI / 180.0, 10, 0, 10);
    Scalar color = Scalar(0, 0, 255);
    for (size_t i = 0; i < plines.size(); i++) {
        Vec4f hline = plines[i];
        line(dst, Point(hline[0], hline[1]), Point(hline[2], hline[3]), color, 3, LINE_AA);
    }

    imshow("效果图", dst);

    waitKey(0);
    return 0;

}

 

 

主要参考博客:

原理部分:http://www.cnblogs.com/php-rearch/p/6760683.html(相当清楚,不解释,)

源码分析1:http://blog.youkuaiyun.com/zhaocj/article/details/50281537(赵春江老师,很多源码都给出了详解,尤其是那篇sift,看的热血沸腾)

源码分析2:http://blog.youkuaiyun.com/traumland/article/details/51319644

源码分析3:http://blog.youkuaiyun.com/sunshine_in_moon/article/details/45271647

Samples1:http://blog.youkuaiyun.com/poem_qianmo/article/details/26977557/   (还是浅墨)

Samples2:http://www.cnblogs.com/skyfsm/p/6881686.html
讲解:https://blog.youkuaiyun.com/qq_37059483/article/details/77891698
 这篇博客的博主在进行Canny之前使用了大津法先进行分割:http://blog.youkuaiyun.com/dcrmg/article/details/52464209

 

园检测

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

const int kvalue = 15;//双边滤波邻域大小

int main()
{
    Mat src_color = imread("E:\\VS2015Opencv\\vs2015\\project\\picture\\08.jpg");//读取原彩色图
    imshow("原图-彩色", src_color);

    //声明一个三通道图像,像素值全为0,用来将霍夫变换检测出的圆画在上面
    Mat dst(src_color.size(), src_color.type());
    dst = Scalar::all(0);

    Mat src_gray;//彩色图像转化成灰度图
    cvtColor(src_color, src_gray, COLOR_BGR2GRAY);
    imshow("原图-灰度", src_gray);
    imwrite("src_gray.png", src_gray);

    Mat bf;//对灰度图像进行双边滤波
    bilateralFilter(src_gray, bf, kvalue, kvalue * 2, kvalue / 2);
    imshow("灰度双边滤波处理", bf);
    imwrite("src_bf.png", bf);

    vector<Vec3f> circles;//声明一个向量,保存检测出的圆的圆心坐标和半径
    HoughCircles(bf, circles, CV_HOUGH_GRADIENT, 1.5, 20, 130, 38, 10, 50);//霍夫变换检测圆

    cout << "x=\ty=\tr=" << endl;
    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(dst, center, 0, Scalar(0, 255, 0), -1, 8, 0);
        circle(dst, center, radius, Scalar(0, 0, 255), 1, 8, 0);

        cout << cvRound(circles[i][0]) << "\t" << cvRound(circles[i][1]) << "\t"
            << cvRound(circles[i][2]) << endl;//在控制台输出圆心坐标和半径                
    }

    imshow("特征提取", dst);
    imwrite("dst.png", dst);

    waitKey();
}

 

 可以参考:http://www.opencv.org.cn/opencvdoc/2.3.2/html/doc/tutorials/imgproc/table_of_content_imgproc/table_of_content_imgproc.html#table-of-content-imgproc

 

转载于:https://www.cnblogs.com/fcfc940503/p/11303533.html

### 使用霍夫变换进行直线检测 #### 霍夫变换简介 霍夫变换是一种用于在图像中识别特定形状的技术,尤其适用于检测简单几何图形如直线形等。该方法通过将原始图像转换到参数空间来实现目标对象的提取[^3]。 #### 直线检测原理 为了理解如何应用霍夫变换来进行直线检测,考虑一个简单的例子:假设有一个二维平面上的一个 (x₀, y₀),那么任何穿过此直线都可以表示为 \(y_0 = kx_0 + b\) 的形式,在这里 \(k\) 是斜率而 \(b\) 则代表截距。当我们将这些关系式看作是在另一个由 \(k\) 和 \(b\) 组成的空间内的单一线条时,则可以通过查找多个这样的线条之间的交叉来定位实际存在于原图中的那条唯一的直线[^4]。 然而,由于垂直线的情况会导致无限大的斜率值,所以通常会采用极坐标系 (\(ρ\),\(θ\)) 来代替传统的笛卡儿坐标系描述直线的位置。此时,每一对 \((x_i , y_i)\) 坐标的组合都会被映射至一系列不同的角度-距离对上形成一条正弦波形轨迹;如果多组数据共同指向同一个位置的话就说明它们属于同一直线上的一部分[^1]。 ```python import cv2 import numpy as np def detect_lines(image_path): img = cv2.imread(image_path, cv2.IMREAD_GRAYSCALE) edges = cv2.Canny(img, 50, 150, apertureSize=3) lines = cv2.HoughLines(edges, rho=1, theta=np.pi/180, threshold=200) result_img = cv2.cvtColor(img, cv2.COLOR_GRAY2BGR) if lines is not None: for line in lines: r,theta=line[0] a,b=(np.cos(theta),np.sin(theta)) x0,y0=a*r,b*r x1=int(x0+1000*(-b)) y1=int(y0+1000*(a)) x2=int(x0-1000*(-b)) y2=int(y0-1000*(a)) cv2.line(result_img,(x1,y1),(x2,y2),(0,0,255),2) return result_img ``` 这段Python代码展示了怎样运用`cv2.HoughLines()`函数执行标准霍夫变换从而找到输入灰度图片中存在的所有可能的直线,并绘制出来显示给用户查看。 #### 检测原理 对于周来说,同样也可以借助类似的思路完成探测工作。具体而言就是在三维参数空间里寻找满足条件的中心以及半径三者间的最佳匹配方案。OpenCV提供了专门针对这种情况优化过的API `HoughCircles()`, 它能够有效地减少计算量并提高准确性[^2]。 ```python def detect_circles(image_path): img = cv2.imread(image_path, cv2.IMREAD_COLOR) gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY) circles=cv2.HoughCircles(gray,cv2.HOUGH_GRADIENT,dp=1,minDist=20,param1=50,param2=30,minRadius=0,maxRadius=0) output=img.copy() if circles is not None: detected_circles=np.uint16(np.around(circles)) for circle in detected_circles[0,:]: center=(circle[0],circle[1]) radius=circle[2] # Draw the circumference of the circle. cv2.circle(output,center,radius,(0,255,0),2) # Draw a small filled circle at the center point to highlight it better visually. cv2.circle(output,center,2,(0,0,255),3) return output ``` 上述脚本实现了基本的功能——读取彩色图像文件作为源素材,接着调用`cv2.HoughCircles()`去发现潜在的目标物体并将结果可视化地呈现出来。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值