美文网首页OpenCvOpenCV
2018-01-15 椭圆检测与拟合

2018-01-15 椭圆检测与拟合

作者: y4nghan | 来源:发表于2018-01-15 10:43 被阅读22次

椭圆检测与拟合

欢迎到YangHan's Notebook 椭圆检测与拟合 获得更佳阅读体验。

实验目标

  • 调⽤CvBox2D cvFitEllipse2( const CvArr* points )实现椭圆拟合

实验环境

  • Windows 10 1709
  • OpenCV 3.3

实验过程

实现了一个fitEllipse()函数,函数原型如下:

void fitEllipse(char* filename, int threshold);

传入图片路径,然后显示出图片椭圆拟合之后的效果。

支持命令行解析图片路径参数。

FitEllipse.exe  test.png

如果没有路径参数,默认时当前目录下的test.png

首先把图片读进来,包括一份灰度图和一份原图。

Mat gray_img = imread(filename, IMREAD_GRAYSCALE);
Mat result = imread(filename);

把灰度图二值化:

Mat binary_img = gray_img >= thresh;

然后使用findContours()检测二值化图像的轮廓点。

findContours(binary_img, contours, RETR_LIST, CHAIN_APPROX_NONE);

其中,参数3可以取值为:

  • RETR_EXTERNEL: 只检测最外围轮廓,包含在外围轮廓内的内围轮廓被忽略
  • RETR_LIST: 检测所有的轮廓,包括内围、外围轮廓,但是检测到的轮廓不建立等级关系,彼此之间独立,没有等级关系,这就意味着这个检索模式下不存在父轮廓或内嵌轮廓
  • RETR_CCOMP: 检测所有的轮廓,但所有轮廓只建立两个等级关系,外围为顶层,若外围内的内围轮廓还包含了其他的轮廓信息,则内围内的所有轮廓均归属于顶层
  • RETR_TREE: 检测所有轮廓,所有轮廓建立一个等级树结构。外层轮廓包含内层轮廓,内层轮廓还可以继续包含内嵌轮廓。

这里我们只选择RETR_LIST即可满足椭圆拟合的要求。

参数4可以取值为:

  • CHAIN_APPROX_NONE 保存物体边界上所有连续的轮廓点到contours向量内
  • CHAIN_APPROX_SIMPLE 仅保存轮廓的拐点信息,把所有轮廓拐点处的点保存入contours向量内,拐点与拐点之间直线段上的信息点不予保留
  • CHAIN_APPROX_TC89_L1, CHAIN_APPROX_TC89_KCOS: 使用teh-Chin Chain 近似算法

这里直接选择简单的CHAIN_APPROX_NONE

然后对于检测出的轮廓点,用椭圆去拟合:

for each (auto contour in contours)
{
  if (contour.size() < 6) continue;
  RotatedRect box = fitEllipse(contour);
  ellipse(result, box, Scalar(0, 255, 255), 1, LINE_AA);
}

椭圆的拟合至少需要6个点,所以把少于6个点的检测结果直接丢弃,然后对于剩下的点用cv2::fitEllipse()来拟合,然后把椭圆绘制在原图上。

之后再保存结果就行了。

实验结果

原图:

test.png

结果:

result.png

可以看到椭圆基本上都检测并拟合出来了。

心得体会

这次实验就是先检测出图像的轮廓点,然后用fitEllipse()函数来拟合椭圆,整体不是太难。然后注意到一点就是imread()读入图片的时候第二个参数可以选择读入的模式,可以用IMREAD_GRAYSCALE让其读入单通道的图片矩阵数据。

附:源代码

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

void fitEllipse(char* filename, int threshold);

int main(int argc, char** argv) {
    char* filename;

    if (argc == 2) {
        filename = argv[1];
    }
    else {
        filename = "test.png";
    }


    fitEllipse(filename, 150);
    cvWaitKey(0);
    destroyAllWindows();

    return 0;
}


void fitEllipse(char* filename, int thresh) {

    Mat gray_img = imread(filename, IMREAD_GRAYSCALE);
    Mat result = imread(filename);
    vector<vector<Point>> contours;
    Mat binary_img = gray_img >= thresh;
    findContours(binary_img, contours, RETR_LIST, CHAIN_APPROX_NONE);
    for each (auto contour in contours)
    {
        if (contour.size() < 6) continue;
        RotatedRect box = fitEllipse(contour);
        ellipse(result, box, Scalar(0, 255, 255), 1, LINE_AA);
    }

    imwrite("result.png", result);
    imshow(filename, gray_img);
    imshow("result", result);
}

相关文章

  • 2018-01-15 椭圆检测与拟合

    椭圆检测与拟合 欢迎到YangHan's Notebook 椭圆检测与拟合 获得更佳阅读体验。 实验目标 调⽤Cv...

  • 椭圆检测

    Hough变换检测椭圆 附带matlab与opencv代码 QT+opencv学习笔记(5)——霍夫直线检测、圆检...

  • Intel-ML笔记02 训练集分割验证&线性规划

    欠拟合(Underfitting)和过拟合(Overfitting) 欠拟合:拟合函数与训练集误差较大过拟合:拟合...

  • 正则化

    一. 欠拟合、正拟合与过拟合 令 二. 正则化——规避过拟合 看过拟合的f(x)形式,即求解“让W向量中...

  • task03

    过拟合、欠拟合及其解决方案 过拟合与欠拟合 欠拟合 过拟合 解决方案 权重衰减 L2 范数正则化 范数正则化在模型...

  • c++ opencv椭圆检测

    1.源码实现 2.编译源码 3.原图及结果

  • 机器学习入门笔记系列(4) | 过拟合与正则化

    欠拟合与过拟合定义 上图表示了拟合数据三种情况:欠拟合(左 2 图)、较好的拟合(左 3 图)、过拟合(左 4 图...

  • 第 7 章 提取直线、轮廓和区域

    本章包括以下内容: 用Canny 算子检测图像轮廓; 用霍夫变换检测直线; 点集的直线拟合; 提取连续区域; 计算...

  • 过拟合与欠拟合及方差偏差

    在模型的评估与调整的过程中,经常会遇到过拟合与欠拟合的情况,如何有效的识别过拟合和欠拟合现象,并了解其中原因,有效...

  • 过拟合的一些问题

    过拟合,欠拟合与模型的容量息息相关。模拟的容量指其拟合数据的能力,容量低的模型难以拟合训练,出现欠拟合,容量高的模...

网友评论

    本文标题:2018-01-15 椭圆检测与拟合

    本文链接:https://www.haomeiwen.com/subject/aaeroxtx.html