【从零学习OpenCV 4】轮廓外接多边形

重磅干货,第一时间送达
经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍《OpenCV 4开发详解》。为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通,提前在公众号上连载部分内容,请持续关注小白。

由于噪声和光照的影响,物体的轮廓会出现不规则的形状,根据不规则的轮廓形状不利于对图像内容进行分析,此时需要将物体的轮廓拟合成规则的几何形状,根据需求可以将图像轮廓拟合成矩形、多边形等。本小节将介绍OpenCV 4中提供的轮廓外接多边形函数,实现图像中轮廓的形状拟合。

矩形是常见的几何形状,矩形的处理和分析方法也较为简单,OpenCV 4提供了两个函数求取轮廓外接矩形,分别是求取轮廓最大外接矩形的boundingRect()函数和求取轮廓最小外接矩形的minAreaRect()函数。

寻找轮廓外接最大矩形就是寻找轮廓X方向和Y方向两端的像素,该矩形长和宽分别与图像的两个轴平行。boundingRect()函数可以实现这个功能,该函数的函数原型在代码清单7-19中给出。

代码清单7-19 boundingRect()函数原型Rect cv::boundingRect(InputArray array)
  • array:输入的灰度图像或者2D点集,数据类型为vector或者Mat。

该函数可以求取包含输入图像中物体轮廓或者2D点集的最大外接矩形,函数只有一个参数,可以是灰度图像或者2D点集,灰度图像的参数类型为Mat,2D点集的参数类型为vector或者Mat。该函数的返回值是一个Rect类型的变量,该变量可以直接用rectangle()函数绘制矩形。返回值共有四个参数,前两个参数是最大外接矩形左上角第一个像素的坐标,后两个参数分别表示最大外接矩形的宽和高。

最小外接矩形的四个边都与轮廓相交,该矩形的旋转角度与轮廓的形状有关,多数情况下矩形的四个边不与图像的两个轴平行。minAreaRect()函数可以求取轮廓的最小外接矩形,该函数的函数原型在代码清单7-20中给出。

代码清单7-20 minAreaRect()函数原型RotatedRect cv::minAreaRect(InputArray points)
  • points:输入的2D点集合

该函数可以根据输入的2D点集合计算最小的外接矩形,函数的返回值是RotatedRect类型的变量,含有矩形的中心位置、矩形的宽和高和矩形旋转的角度。RotatedRect类具有两个重要的方法和属性,可以输出矩形的四个顶点和中心坐标。输出四个顶点坐标的方法是points(),假设RotatedRect类的变量为rrect,可以通过rrect.points(points)命令进行读取,其中坐标存放的变量是Point2f类型的数组。输出矩形中心坐标的属性是center,假设RotatedRect类的变量为rrect,可以通过opt=rrect.center命令进行读取,其中坐标存放的变量是Point2f类型的变量。

为了了解两个外接矩形函数的使用方法,代码清单7-21中给出了提取轮廓外接矩形的示例程序。程序中首先利用Canny算法提取图像边缘,之后通过膨胀算法将邻近的边缘连接成一个连通域,然后提取图像的轮廓,并提取每一个轮廓的最大外接矩形和最小外接矩形,最后在图像中绘制出矩形轮廓,程序的运行结果在图7-20给出。

代码清单7-21 myRect.cpp计算轮廓外接矩形#include <opencv2/opencv.hpp>#include <iostream>#include <vector>
using namespace cv;using namespace std;
int main(){Mat img = imread("stuff.jpg");if (img.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}Mat img1, img2;img.copyTo(img1); //深拷贝用来绘制最大外接矩形img.copyTo(img2); //深拷贝迎来绘制最小外接矩形imshow("img", img);
// 去噪声与二值化Mat canny;Canny(img, canny, 80, 160, 3, false);imshow("", canny);
//膨胀运算,将细小缝隙填补上Mat kernel = getStructuringElement(0, Size(3, 3));dilate(canny, canny, kernel);
// 轮廓发现与绘制vector<vector<Point>> contours;vector<Vec4i> hierarchy;findContours(canny, contours, hierarchy, 0, 2, Point());
//寻找轮廓的外接矩形for (int n = 0; n < contours.size(); n++){// 最大外接矩形Rect rect = boundingRect(contours[n]);rectangle(img1, rect, Scalar(0, 0, 255), 2, 8, 0);
// 最小外接矩形RotatedRect rrect = minAreaRect(contours[n]);Point2f points[4];rrect.points(points); //读取最小外接矩形的四个顶点Point2f cpt = rrect.center; //最小外接矩形的中心
// 绘制旋转矩形与中心位置for (int i = 0; i < 4; i++){if (i == 3){line(img2, points[i], points[0], Scalar(0, 255, 0), 2, 8, 0);break;}line(img2, points[i], points[i + 1], Scalar(0, 255, 0), 2, 8, 0);}//绘制矩形的中心circle(img, cpt, 2, Scalar(255, 0, 0), 2, 8, 0);}
//输出绘制外接矩形的结果imshow("max", img1);imshow("min", img2);waitKey(0);return 0;}

图7-20 myRect.cpp程序运行结果

有时候用矩形逼近轮廓会造成较大的误差,例如图7-20中对于圆形轮廓的逼近矩形围成的面积比真实轮廓面积大,如果寻找逼近轮廓的多边形,那么多边形围成的面积会更加接近真实的圆形轮廓面积。OpenCV 4提供了approxPolyDP()函数用于寻找逼近轮廓的多边形,该函数的函数原型在代码清单7-22中给出。

代码清单7-22 approxPolyDP()函数原型void cv::approxPolyDP(InputArray curve, OutputArray approxCurve, double epsilon, bool closed )
  • curve:输入轮廓像素点。
  • approxCurve:多边形逼近结果,以多边形顶点坐标的形式给出。
  • epsilon:逼近的精度,即原始曲线和逼近曲线之间的最大距离。
  • closed:逼近曲线是否为封闭曲线的标志, true表示曲线封闭,即最后一个顶点与第一个顶点相连。

该函数根据输入的轮廓得到最佳的逼近多边形。函数的第一个参数是输入的轮廓2D像素点,数据类型是vector或者Mat。第二个参数是多边形的逼近结果,以多边形顶点坐标的形式输出,是CV_32SC2类型的N×1的Mat类矩阵,可以通过输出结果的顶点数目初步判断轮廓的几何形状。第三个参数是多边形逼近时的精度,即原始曲线和逼近曲线之间的最大距离。第四个参数是逼近曲线是否为封闭曲线的标志, true表示曲线封闭,即最后一个顶点与第一个顶点相连。

为了了解该函数用法,在代码清单7-23中给出了对多个轮廓进行多边形逼近的示例程序。程序中首先提取了图像的边缘,然后对边缘进行腐蚀运算将靠近的边缘变成一个连通域,之后对边缘结果进行轮廓检测,并对每个轮廓进行多边形逼近,将逼近结果绘制在原图像中,并通过判断逼近多边形的顶点数目识别轮廓的形状,程序运行结果在图7-21给出。

代码清单7-23 myApproxPolyDP.cpp多边形轮廓拟合#include <opencv2/opencv.hpp>#include <iostream>#include <vector>
using namespace cv;using namespace std;
//绘制轮廓函数void drawapp(Mat result, Mat img2){for (int i = 0; i < result.rows; i++){//最后一个坐标点与第一个坐标点连接if (i == result.rows - 1){Vec2i point1 = result.at<Vec2i>(i);Vec2i point2 = result.at<Vec2i>(0);line(img2, point1, point2, Scalar(0, 0, 255), 2, 8, 0);break;}Vec2i point1 = result.at<Vec2i>(i);Vec2i point2 = result.at<Vec2i>(i + 1);line(img2, point1, point2, Scalar(0, 0, 255), 2, 8, 0);}}
int main(int argc, const char *argv[]){Mat img = imread("approx.png");if (img.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}// 边缘检测Mat canny;Canny(img, canny, 80, 160, 3, false);//膨胀运算Mat kernel = getStructuringElement(0, Size(3, 3));dilate(canny, canny, kernel);
// 轮廓发现与绘制vector<vector<Point>> contours;vector<Vec4i> hierarchy;findContours(canny, contours, hierarchy, 0, 2, Point());
//绘制多边形for (int t = 0; t < contours.size(); t++){//用最小外接矩形求取轮廓中心RotatedRect rrect = minAreaRect(contours[t]);Point2f center = rrect.center;circle(img, center, 2, Scalar(0, 255, 0), 2, 8, 0);
Mat result;approxPolyDP(contours[t], result, 4, true); //多边形拟合drawapp(result, img);cout << "corners : " << result.rows << endl;
//判断形状和绘制轮廓if (result.rows == 3){putText(img, "triangle", center, 0, 1, Scalar(0, 255, 0), 1, 8);}if (result.rows == 4){putText(img, "rectangle", center, 0, 1, Scalar(0, 255, 0), 1, 8);}if (result.rows == 8){putText(img, "poly-8", center, 0, 1, Scalar(0, 255, 0), 1, 8);}if (result.rows > 12){putText(img, "circle", center, 0, 1, Scalar(0, 255, 0), 1, 8);}}imshow("result", img);waitKey(0);return 0;}

图7-21 myApproxPolyDP.cpp程序中多边形拟合结果

(0)

相关推荐

  • OpenCV探索之路(十一):轮廓查找和多边形包围轮廓

    Canny一类的边缘检测算法可以根据像素之间的差异,检测出轮廓边界的像素,但它没有将轮廓作为一个整体.所以要将轮廓提起出来,就必须将这些边缘像素组装成轮廓. OpenCV中有一个很强大的函数,它可以从 ...

  • OpenCV 3.4.11 cv::dnn::Net::forward()函数第一个参数的理解

    问题来源 使用OpenCV 3.4.11在C++下跑YOLOv4的时候对cv::dnn::Net::forward()函数的第一个参数产生了一些疑问,在此记录学习解惑的过程 代码来源: https:/ ...

  • opencv笔记(二十九)——提取轮廓相关函数使用方法

    opencv中常用的跟轮廓相关的操作有:findContours()查找轮廓:drawContours()画轮廓:轮廓填充:计算轮廓的面积和周长:提取轮廓凸包,矩形,最小外接矩形,外接圆等.它们都有相 ...

  • OpenCV学习28

    查找轮廓 什么是轮廓:一个轮廓是由图像中的一系列点组成的,也就是图像中的一条曲线.在OpenCV中一般用序列来存储轮廓信息.序列中的每个元素是曲线中每个点的位置. 关于序列:序列是内存存储器中可以存储 ...

  • 基于OpenCV的dnn模块使用YOLOv3进行目标检测

    0.说明: 测试的opencv版本为opencv3.4.5 电脑cup:intel 4代i5(4200U) 1.YOLO介绍: YOLO详解(知乎) 2.下载yolov3的配置文件: wget htt ...

  • OpenCV基础知识入门

    本文旨在让你快速入门opencv. OpenCV OpenCV是计算机视觉中最受欢迎的库,最初由intel使用C和C ++进行开发的,现在也可以在python中使用.该库是一个跨平台的开源库,是免费使 ...

  • 【从零学习OpenCV 4】轮廓发现与绘制

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】深度神经网络应用实例

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】图像修复

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】分割图像——Mean-Shift分割算法

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<从零学习OpenCV 4>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】分割图像——Grabcut图像分割

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】分割图像——分水岭法

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】QR二维码检测

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】图像矩的计算与应用

    重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<从零学习OpenCV 4>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版社沟通 ...

  • 【从零学习OpenCV 4】直线拟合

    ‍‍ 重磅干货,第一时间送达 经过几个月的努力,小白终于完成了市面上第一本OpenCV 4入门书籍<OpenCV 4开发详解>.为了更让小伙伴更早的了解最新版的OpenCV 4,小白与出版 ...