基础概念大概了解,可以在看完详细解释之后再回过头来看基础概念,相信会有进一步的深入了解。
图中有五块颜色区域,分别记为A,B,C,D,E,每块区域的外部边界和内部边界都各自组成轮廓,轮廓数如下图所示:
构建轮廓树为:
每一个节点就是一个轮廓,根据每个节点在层次中的四元数组索引(如下表所示),图中都进行了标记。
索引 | 含义 |
---|---|
0 | 同级的下一条轮廓 |
1 | 同级的上一条轮廓 |
2 | 下级的第一个子节点 |
3 | 上级的父节点 |
函数原型:
findContours( InputOutputArray image, OutputArrayOfArrays contours,
OutputArray hierarchy, int mode,
int method, Point offset=Point());
参数含义:
vector> contours;
向量,向量内每个元素保存了一组由连续的Point点构成的点的集合的向量,每一组Point点集就是一个轮廓,有多少轮廓,向量contours就有多少元素;vector hierarchy;
,表示向量内每一个元素包含了4个int型变量——hierarchy[i][0] —hierarchy[i][3],分别表示第i个轮廓的后一个轮廓、前一个轮廓、第一条子轮廓、父轮廓的索引编号,如果当前轮廓没有对应的后一个轮廓,前一个轮廓、第一条子轮廓或父嵌轮廓的话,则hierarchy[i][0] —hierarchy[i][3]的相应位被设置为默认值-1;先看一段简单的代码,之后会进行详细介绍:
vector<vector<Point>> contours;
vector<Vec4i> hierarchy;
findContours(image, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point());
函数原型:
drawContours(InputOutputArray image, InputArrayOfArrays contours,
int contourIdx, const Scalar& color, int thickness=1, int lineType=8,
InputArray hierarchy=noArray(), int maxLevel=INT_MAX, Point offset=Point() )
参数含义:
绘制轮廓部分代码如下:
Mat imageContours = Mat::zeros(image.size(), CV_8UC1);
for (int i = 0; i < contours.size(); i++)
{ //绘制轮廓
drawContours(imageContours, contours, i, Scalar(255), 1, 8, hierarchy);
}
基本流程如下:
①将输出图像使用cvtColor转换为灰度图像
②使用边缘算法(如Canny)进行边缘提取
③使用findContours寻找轮廓
④使用drawContours绘制轮廓
完整程序如下:
#include
#include "iostream"
using namespace std;
using namespace cv;
int main(int argc, char *argv[])
{
Mat imageSource = imread("D:/Desktop/1.png", 0);
imshow("Source Image", imageSource);//灰度图像
Mat image;
GaussianBlur(imageSource, image, Size(3, 3), 0);//高斯滤波
Canny(image, image, 100, 250);//canny算子边缘检测
vector<vector<Point>> contours;
vector<Vec4i> hierarchy;
findContours(image, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point());
Mat imageContours = Mat::zeros(image.size(), CV_8UC1);
Mat Contours = Mat::zeros(image.size(), CV_8UC1); //绘制
for (int i = 0; i < contours.size(); i++)
{
//contours[i]代表的是第i个轮廓,contours[i].size()代表的是第i个轮廓上所有的像素点数
for (int j = 0; j < contours[i].size(); j++)
{ //绘制出contours向量内所有的像素点
Point P = Point(contours[i][j].x, contours[i][j].y);
Contours.at<uchar>(P) = 255;
}
输出hierarchy向量内容
cout << "向量hierarchy的第" << i << "个元素内容为:" << endl << hierarchy[i] << endl << endl;
//绘制轮廓
drawContours(imageContours, contours, i, Scalar(255), 1, 8, hierarchy);
}
imshow("Contours Image", imageContours); //轮廓
imshow("Point of Contours", Contours); //向量contours内保存的所有轮廓点集
waitKey(0);
return 0;
}
原图像:
灰度化图像:
边缘轮廓绘制后的图像:
向量contours内保存的所有轮廓点集形成的图像:
这种参数表示:只检测最外层轮廓。
边缘轮廓绘制后的图像:
可以发现,只有最外层的轮廓被检测到,内层的轮廓被自动忽略。
向量contours内保存的所有轮廓点集形成的图像:
可以发现,点集保存了所有轮廓上的所有点,图像表现跟轮廓一致。
hierarchy向量输出内容为:
hierarchy向量中每个元素的四个整型分别表示当前轮廓的后一个轮廓,前一个轮廓,第一条子轮廓和父轮廓的索引编号。
本次参数配置下,hierarchy向量内有4个元素,分别对应于4个轮廓。以第2个轮廓(对应向量内第1个元素)为例,内容为[2,0,-1,-1], “2”表示当前轮廓的后一个轮廓的编号为2,“0”表示当前轮廓的前一个轮廓编号为0,其后2个“-1”表示为空,因为只有最外层轮廓这一个等级,所以不存在父轮廓和内嵌轮廓。
这类参数表示:检测所有轮廓,但各轮廓之间彼此独立。
向量contours内保存的所有轮廓点集形成的图像:
可以发现,点集中直线段部分都被省略掉了,其他部分得到了保留。
这类参数表示:检测所有轮廓,并且轮廓间建立外层、内层的等级关系,并且保存轮廓上所有点。
向量contours内保存的所有轮廓点集形成的图像:
contours点集组成的图形跟轮廓表现一致。
hierarchy向量输出内容为:
可以观察到,每个轮廓都被划分等级,最外围、第一内围、第二内围等等,所以除第1个最后一个轮廓外,其他轮廓都具有不为-1的第3、第4个整形参数。
如果对你有所帮助,记得点个赞哟~
参考:
1.《学习OpenCV3》
2.OpenCV中的findContours函数参数详解