OpenCV-霍夫变换

OpenCV-霍夫变换


author@jason_ql
http://blog.youkuaiyun.com/lql0716/


  • 标准霍夫变换(Standard Hough Transform, SHT)
  • 多尺度霍夫变换(Multi—Scale Hough Transform, MSHT)
  • 累计概率霍夫变换(Progressive Probabilistic Hough Transform,PPHT)

相关函数

  • HoughLines()直线检测
    该函数对应标准霍夫变换(SHT)、多尺度霍夫变换(MSHT)。

C++: void HoughLines(InputArray image, OutputArray lines, double rho, double theta, int threshold, double srn=0, double stn=0)

第一个参数:InputaArray类型的image,原图像,8位单通道二进制。
第二个参数:InputArray类型的lines,经过调用HoughLines函数后储存了霍夫变换检测到线条的输出矢量。每一条线由具有两个元素的矢量(rou,theta)表示,其中rou为到坐标原点(0,0)(原点为图像的左上角)的距离,theta是弧度线条旋转角度(0度表示垂直线,pi/2度表示水平线)。
第三个参数:double类型的rho,以像素为单位的距离精度。另一种表述方式是直线搜索时的进步尺寸的单位角度。
第四个参数:double类型的theta,以弧度为单位的角度精度。另一种表述方式是直线搜索时的进步尺寸的单位角度。
第五个参数:int类型的threshold,累加平面的阈值参数,即识别某部分为图中的一条直线时它在累加平面中必须达到的值。大于阈值threshold的线段才可以被检测通过并返回到结果中。
第六个参数:double类型的srn,默认值为0.对于多尺度的霍夫变换,这是第三个参数进步尺寸rho的除数距离。粗略的累加器进步尺寸直接是第三个参数rho,而精确的累加器进步尺寸为rho/srn。
第七个参数:double类型的stn,默认值为0.对于多尺度霍夫变换,srn表示第四个参数进步尺寸的单位角度theta的除数距离。且如果srn和stn同时为0,就表示使用经典的霍夫变换,否则这两个参数应该都为正数。

  • HoughLinesP() 直线检测
    该函数对应累计概率霍夫变换PPHT。

  • HoughCircles() 圆检测

  • 霍夫变换示例代码

#include <opencv2/opencv.hpp>

using namespace cv;
using namespace std;

int main()
{
//    String path = "/home/jason/jason2/photo/cylinder/target_cylinder.jpg";
    String path = "/home/jason/jason2/photo/cylinder/01.jpg";
    cv::Mat img1, img2, gray1, gray2;
    img1 = cv::imread(path);
    cv::cvtColor(img1, gray1, cv::COLOR_BGR2GRAY);

    cv::Canny(img1, img2, 50, 100);
//    cv::cvtColor(img2, gray2, cv::COLOR_BGR2GRAY);

    cv::vector<Vec2f> lines;
    cv::HoughLines(img2, lines, 1, CV_PI/180, 150, 0, 0);

    cv::Point v0;
    v0.x = 0;
    v0.y = 1;

    for(size_t i = 0; i < lines.size(); i++)
    {
        float rho = lines[i][0], theta = lines[i][1];
        cv::Point pt1, pt2, v1;
        double a = cos(theta), b = sin(theta);
        double x0 = a*rho, y0 = b*rho;

        pt1.x = round(x0 + 1000 * (-b));
        pt1.y = round(y0 + 1000 * a);
        pt2.x = round(x0 - 1000 * (-b));
        pt2.y = round(y0 - 1000 * a);

        v1 = pt1 - pt2;

        if( (v0.x * v1.y + v0.y * v1.x) == 0)
        {
            line(img1, pt1, pt2, Scalar(0, 255, 0), 2, CV_AA);
        }

    }

    cv::namedWindow("原图", cv::WINDOW_NORMAL);
    cv::namedWindow("直线", cv::WINDOW_NORMAL);
    cv::imshow("原图", img1);
    cv::imshow("直线", img2);

    cv::waitKey(0);
    return 0;

}
#霍夫变换圆检测

#include <cv.h>
#include <highgui.h>
#include <math.h>

using namespace cv;

int main(int argc, char** argv)
{
    Mat img, gray;
    if( argc != 2 && !(img=imread(argv[1], 1)).data)
        return -1;
    cvtColor(img, gray, CV_BGR2GRAY);
    // smooth it, otherwise a lot of false circles may be detected
    GaussianBlur( gray, gray, Size(9, 9), 2, 2 );
    vector<Vec3f> circles;
    HoughCircles(gray, circles, CV_HOUGH_GRADIENT,
                 2, gray->rows/4, 200, 100 );
    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]);
         // draw the circle center
         circle( img, center, 3, Scalar(0,255,0), -1, 8, 0 );
         // draw the circle outline
         circle( img, center, radius, Scalar(0,0,255), 3, 8, 0 );
    }
    namedWindow( "circles", 1 );
    imshow( "circles", img );
    return 0;
}

  • 相关文章
<think>好的,用户想用OpenCV-Python实现霍夫变换来检测图像中的矩形。首先,我需要回忆一下霍夫变换的相关知识。霍夫变换通常用于检测直线和圆形,检测矩形可能需要结合直线检测和一些后处理步骤。 用户提供的参考内容里提到了轮廓检测和直线检测,比如使用cv2.findContours和cv2.HoughLines。这可能对解决问题有帮助。首先,应该考虑图像预处理,比如灰度化、边缘检测。Canny边缘检测是常用的方法,参考内容里也有提到。 接下来,使用霍夫直线检测来找到图像中的直线。OpenCV的HoughLines或HoughLinesP函数可以检测直线,其中HoughLinesP是概率霍夫变换,效率更高,适合实时应用。检测到直线后,需要将这些直线组合成矩形。这可能需要计算直线的交点,或者寻找四边形的轮廓。 不过,用户提到的是检测矩形,或许另一种方法是先检测所有轮廓,然后筛选出四边形的轮廓,并进一步验证其是否为矩形。参考内容中的cv2.findContours和轮廓近似方法可能适用。比如,使用cv2.approxPolyDP来近似轮廓,如果近似后的轮廓有四个顶点,且角度接近直角,则可以认为是矩形。 需要结合两种方法:霍夫变换检测直线,然后找出四条直线形成矩形;或者使用轮廓检测直接找四边形。用户明确提到霍夫变换,所以可能需要优先考虑直线检测的方法。但霍夫变换检测直线后如何组合成矩形,这可能需要计算直线的交点和角度,找出相互垂直且闭合的四边形。 同时,注意图像预处理的重要性,比如降噪、二值化等,参考内容里提到使用二值化来提取硬币轮廓,可能同样适用于矩形检测。此外,处理后的边缘图像质量直接影响霍夫变换的效果。 然后,编写代码步骤可能包括:读取图像,转为灰度,高斯模糊,Canny边缘检测,霍夫变换检测直线,处理直线数据,寻找四个边组成矩形,绘制结果。或者,检测所有直线后,通过筛选水平和垂直的直线,再组合成矩形。 需要注意的是,霍夫变换可能会检测到很多直线,如何筛选出构成矩形的四条是一个挑战。可能需要计算直线之间的交点和角度关系,或者使用其他几何条件来过滤。 另外,用户提供的引用中提到cv.RETR_EXTERNAL和轮廓填充,这可能与轮廓检测方法相关。如果使用轮廓检测,步骤可能更简单:找到轮廓,近似多边形,筛选四个点的轮廓,计算角度是否为直角,面积是否足够大等。 不过用户明确要求使用霍夫变换,所以可能需要将两种方法结合,比如用霍夫检测直线,然后找到四个直线形成矩形,或者用霍夫检测矩形的方法。不过OpenCV中是否有直接的霍夫矩形检测函数?我记得HoughLines和HoughLinesP主要用于直线,而矩形检测可能需要其他处理。 综合以上思考,解决方案可能是:预处理图像,检测边缘,使用霍夫变换检测直线,然后通过直线之间的几何关系筛选出可能的矩形边,最后验证并绘制矩形。或者,检测直线后,寻找闭合的四边形。 现在需要将这些步骤转化为具体的代码示例,并确保正确引用用户提供的参考资料中的方法,比如使用cv2.findContours中的参数,或者边缘检测的处理步骤。同时,注意回答的结构,按照用户的要求,分步骤说明,并生成相关问题。</think>以下是用OpenCV-Python实现霍夫变换检测矩形的分步说明: ### 霍夫变换检测矩形步骤 1. **图像预处理** ```python import cv2 import numpy as np # 读取图像并转为灰度 img = cv2.imread('rect.jpg') gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY) # 高斯模糊降噪 blur = cv2.GaussianBlur(gray, (5,5), 0) # Canny边缘检测(参数根据实际情况调整) edges = cv2.Canny(blur, 50, 150) ``` 2. **霍夫直线检测** ```python # 概率霍夫变换检测直线 lines = cv2.HoughLinesP(edges, rho=1, theta=np.pi/180, threshold=50, minLineLength=50, maxLineGap=10) # 绘制所有检测到的直线 if lines is not None: for line in lines: x1,y1,x2,y2 = line[0] cv2.line(img, (x1,y1), (x2,y2), (0,255,0), 2) ``` 3. **直线后处理(关键步骤)** - **筛选水平和垂直线**:计算直线角度,保留0°、90°、45°等特定角度的直线 - **寻找交点**:通过直线方程计算交点,组合成四边形 - **几何验证**:验证四边形内角是否接近90°,边长比例是否符合矩形特征 4. **替代方案:轮廓检测法** ```python # 查找轮廓并筛选矩形 contours, _ = cv2.findContours(edges, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE) for cnt in contours: # 轮廓近似 epsilon = 0.02*cv2.arcLength(cnt,True) approx = cv2.approxPolyDP(cnt,epsilon,True) # 筛选四边形 if len(approx) == 4: # 计算角度验证直角 cv2.drawContours(img, [approx], 0, (0,0,255), 3) ``` 5. **结果显示** ```python cv2.imshow('Result', img) cv2.waitKey(0) cv2.destroyAllWindows() ``` > 注意:实际应用中需要根据具体图像调整参数,建议结合两种方法: > 1. 先用霍夫变换检测直线 > 2. 再用轮廓分析验证几何特征[^2]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

AI人工智能科学

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值