要先变为二值图像:cvThreshold

提取轮廓:cvFindContours

参数描述:

hiararchy:参数和轮廓个数相同。

每个轮廓contours[ i ] 对应4个hierarchy元素的索引编号,即:

    • hierarchy[ i ][ 0 ] 后一个轮廓
    • hierarchy[ i ][ 1 ] 前一个轮廓
    • hierarchy[ i ][ 2 ] 父轮廓
    • hierarchy[ i ][ 3 ] 内嵌轮廓

如果没有对应项,该值设置为负数。

mode:表示轮廓的检索模式

CV_RETR_EXTERNAL      表示只检测外轮廓

CV_RETR_LIST              检测的轮廓不建立等级关系

CV_RETR_CCOMP          建立两个等级的轮廓,上面的一层为外边界,里面的一层为内孔的边界信息。如果内孔内还有一个连通物体,这个物体的边界也在顶层。

CV_RETR_TREE             建立一个等级树结构的轮廓。

method:为轮廓的近似办法

CV_CHAIN_APPROX_NONE         存储所有的轮廓点,相邻的两个点的像素位置差不超过1,即max(abs(x1-x2),abs(y2-y1))==1

CV_CHAIN_APPROX_SIMPLE       压缩水平方向,垂直方向,对角线方向的元素,只保留该方向的终点坐标,例如一个矩形轮廓只需4个点来保存轮廓信息

CV_CHAIN_APPROX_TC89_L1,CV_CHAIN_APPROX_TC89_KCOS       使用teh-Chinl chain 近似算法

offset:表示代表轮廓点的偏移量,可以设置为任意值。对ROI图像中找出的轮廓,并要在整个图像中进行分析时,这个参数还是很有用的。

findContours 后会对输入的二值图像改变,最好需创建新MAT来存放;

findContours 后的轮廓信息contours可能过于复杂不平滑,可以用 approxPolyDP() 对该多边形曲线做适当近似。

contourArea() 函数可以得到当前轮廓包含区域的大小,方便轮廓的筛选。

findContours经常与 drawContours() 配合使用,用来将轮廓绘制出来。

  • 第一个参数,image表示目标图像
  • 第二个参数,contours表示输入的轮廓组,每一组轮廓由点vector构成
  • 第三个参数,contourIdx指明画第几个轮廓,如果该参数为负值,则画全部轮廓
  • 第四个参数,color为轮廓的颜色
  • 第五个参数,thickness为轮廓的线宽,如果为负值或CV_FILLED表示填充轮廓内部
  • 第六个参数,lineType为线型
  • 第七个参数,为轮廓结构信息
  • 第八个参数,为maxLevel

得到了复杂轮廓往往不适合特征的检测,这里再介绍一个点集凸包络的提取函数convexHull()

  • 输入参数,是contours组中的一个轮廓
  • 返回,外凸包络的点集。

还可以得到轮廓的外包络矩形,使用函数 boundingRect()

如果想得到旋转的外包络矩形,使用函数 minAreaRect(),返回值为RotatedRect;

也可以得到轮廓的外包络,对应的函数为 minEnclosingCircle()

想得到轮廓的外包络椭圆,对应的函数为 fitEllipse(),返回值也是RotatedRect,可以用ellipse函数画出对应的椭圆。

如果想根据多边形的轮廓信息 => 多边形的多阶矩,可以使用 类moments,这个类可以得到多边形光栅形状的3阶以内的所有矩,

类内有变量m00,m10,m01,m20,m11,m02,m30,m21,m12,m03,

比如多边形的质心为 x = m10 / m00,y = m01 / m00。

如果想获得一点与多边形封闭轮廓的信息,可以调用 pointPolygonTest(),这个函数返回值为该点距离轮廓最近边界的距离,为正值为在轮廓内部,负值为在轮廓外部,0表示在边界上。

#include "opencv2/imgproc/imgproc.hpp"
#include "opencv2/highgui/highgui.hpp"
#include <math.h>
#include <iostream> using namespace cv;
using namespace std; static void help()
{
cout
<< "\nThis program illustrates the use of findContours and drawContours\n"
<< "The original image is put up along with the image of drawn contours\n"
<< "Usage:\n"
<< "./contours2\n"
<< "\nA trackbar is put up which controls the contour level from -3 to 3\n"
<< endl;
} const int w = 500;
int levels = 3; vector<vector<Point> > contours;
vector<Vec4i> hierarchy; static void on_trackbar(int, void*)
{
Mat cnt_img = Mat::zeros(w, w, CV_8UC3);
int _levels = levels - 3;
drawContours( cnt_img, contours, _levels <= 0 ? 3 : -1, Scalar(128,255,255),
3, LINE_AA, hierarchy, std::abs(_levels) ); imshow("contours", cnt_img);
} int main( int argc, char** argv)
{
cv::CommandLineParser parser(argc, argv, "{help h||}");
if (parser.has("help"))
{
help();
return 0;
}
// Mat img = Mat::zeros(w, w, CV_8UC1); //Jeff --> we don't need to draw this by ourselves.
//Draw 6 faces
// for( int i = 0; i < 6; i++ )
// {
// int dx = (i%2)*250 - 30;
// int dy = (i/2)*150;
// const Scalar white = Scalar(255);
// const Scalar black = Scalar(0); // if( i == 0 )
// {
// for( int j = 0; j <= 10; j++ )
// {
// double angle = (j+5)*CV_PI/21;
// line(img, Point(cvRound(dx+100+j*10-80*cos(angle)),
// cvRound(dy+100-90*sin(angle))),
// Point(cvRound(dx+100+j*10-30*cos(angle)),
// cvRound(dy+100-30*sin(angle))), white, 1, 8, 0);
// }
// } // ellipse( img, Point(dx+150, dy+100), Size(100,70), 0, 0, 360, white, -1, 8, 0 );
// ellipse( img, Point(dx+115, dy+70), Size(30,20), 0, 0, 360, black, -1, 8, 0 );
// ellipse( img, Point(dx+185, dy+70), Size(30,20), 0, 0, 360, black, -1, 8, 0 );
// ellipse( img, Point(dx+115, dy+70), Size(15,15), 0, 0, 360, white, -1, 8, 0 );
// ellipse( img, Point(dx+185, dy+70), Size(15,15), 0, 0, 360, white, -1, 8, 0 );
// ellipse( img, Point(dx+115, dy+70), Size(5,5), 0, 0, 360, black, -1, 8, 0 );
// ellipse( img, Point(dx+185, dy+70), Size(5,5), 0, 0, 360, black, -1, 8, 0 );
// ellipse( img, Point(dx+150, dy+100), Size(10,5), 0, 0, 360, black, -1, 8, 0 );
// ellipse( img, Point(dx+150, dy+150), Size(40,10), 0, 0, 360, black, -1, 8, 0 );
// ellipse( img, Point(dx+27, dy+100), Size(20,35), 0, 0, 360, white, -1, 8, 0 );
// ellipse( img, Point(dx+273, dy+100), Size(20,35), 0, 0, 360, white, -1, 8, 0 );
// } Mat img = imread("/home/unsw/lolo.jpg");
Mat gray;
cvtColor(img, gray, COLOR_RGB2GRAY );
Mat binary;
threshold(gray, binary, 200,255,THRESH_BINARY); // (1) Pic One Show: show the faces
namedWindow( "image", 1 );
imshow( "image", binary ); // (2) Pic Two Show
//Extract the contours so that
vector<vector<Point> > contours0;
findContours( binary, contours0, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE); contours.resize(contours0.size());
for( size_t k = 0; k < contours0.size(); k++ )
approxPolyDP(Mat(contours0[k]), contours[k], 3, true); // Jeff --> the same name to bind window and trackbar together.
// qt qml to draw would be much better.
// callback: on_trackbar()
namedWindow( "contours", 1 );
createTrackbar( "levels+3", "contours", &levels, 7, on_trackbar );
on_trackbar(0,0); waitKey();
return 0;
}

  


Reference: http://blog.csdn.net/felix86/article/details/38121959

采用cvFindContours提取轮廓,并过滤掉小面积轮廓,最后将轮廓保存。

 static int getContoursByCplus(char* Imgname, double minarea, double whRatio)
{
cv::Mat src, dst, canny_output;
/// Load source image and convert it to gray
src = imread(Imgname, ); if (!src.data)
{
std::cout << "read data error!" << std::endl;
return -;
}
blur(src, src, Size(, )); //the pram. for findContours,
vector<vector<Point> > contours;
vector<Vec4i> hierarchy; /// Detect edges using canny
Canny(src, canny_output, , , );
/// Find contours
findContours(canny_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(, ));
//CV_RETR_CCOMP, CV_CHAIN_APPROX_SIMPLE double maxarea = ;
int maxAreaIdx = ; for (int i = ; i<contours.size(); i++)
{ double tmparea = fabs(contourArea(contours[i]));
if (tmparea>maxarea)
{
maxarea = tmparea;
maxAreaIdx = i;
continue;
} if (tmparea < minarea)
{
// *** 删除面积小于设定值的轮廓
contours.erase(contours.begin() + i);
std::wcout << "delete a small area" << std::endl;
continue;
}
//计算轮廓的直径宽高
Rect aRect =boundingRect(contours[i]);
if ((aRect.width / aRect.height)<whRatio)
{
// *** 删除宽高比例小于设定值的轮廓
contours.erase(contours.begin() + i);
std::wcout << "delete a unnomalRatio area" << std::endl;
continue;
}
}
/// Draw contours,彩色轮廓
dst= Mat::zeros(canny_output.size(), CV_8UC3);
for (int i = ; i< contours.size(); i++)
{
//随机颜色
Scalar color = Scalar(rng.uniform(, ), rng.uniform(, ), rng.uniform(, ));
drawContours(dst, contours, i, color, , , hierarchy, , Point());
}
// Create Window
char* source_window = "countors";
namedWindow(source_window, CV_WINDOW_NORMAL);
imshow(source_window, dst);
cv:; waitKey(); return ;
}

[OpenCV] Samples 04: contours2的更多相关文章

  1. [OpenCV] Samples 16: Decompose and Analyse RGB channels

    物体的颜色特征决定了灰度处理不是万能,对RGB分别处理具有相当的意义. #include <iostream> #include <stdio.h> #include &quo ...

  2. [OpenCV] Samples 10: imagelist_creator

    yaml写法的简单例子.将 $ ./ 1 2 3 4 5 命令的参数(代表图片地址)写入yaml中. 写yaml文件. 参考:[OpenCV] Samples 06: [ML] logistic re ...

  3. [OpenCV] Samples 06: [ML] logistic regression

    logistic regression,这个算法只能解决简单的线性二分类,在众多的机器学习分类算法中并不出众,但它能被改进为多分类,并换了另外一个名字softmax, 这可是深度学习中响当当的分类算法 ...

  4. [OpenCV] Samples 06: logistic regression

    logistic regression,这个算法只能解决简单的线性二分类,在众多的机器学习分类算法中并不出众,但它能被改进为多分类,并换了另外一个名字softmax, 这可是深度学习中响当当的分类算法 ...

  5. [OpenCV] Samples 13: opencv_version

    cv::CommandLineParser的使用. I suppose CommandLineParser::has("something") should be true whe ...

  6. [OpenCV] Samples 12: laplace

    先模糊再laplace,也可以替换为sobel等. 变换效果后录成视频,挺好玩. #include "opencv2/videoio/videoio.hpp" #include & ...

  7. [OpenCV] Samples 05: convexhull

    得到了复杂轮廓往往不适合特征的检测,这里再介绍一个点集凸包络的提取函数convexHull,输入参数就可以是contours组中的一个轮廓,返回外凸包络的点集 ---- 如此就能去掉凹进去的边. 对于 ...

  8. [OpenCV] Samples 03: cout_mat

    操作Mat元素时:I.at<double>(1,1) = CV_PI; /* * * cvout_sample just demonstrates the serial out capab ...

  9. [OpenCV] Samples 02: [ML] kmeans

    注意Mat作为kmeans的参数的含义. 扩展:高维向量的聚类. #include "opencv2/highgui.hpp" #include "opencv2/cor ...

随机推荐

  1. yii2更改面包屑的首页链接

    <?= Breadcrumbs::widget([ 'homeLink' => ['label' => 'Home', 'url' => Yii::$app->getHo ...

  2. 地图、定位 CLLocationManager CLGeocoder CLPlacemark

    地图.定位 一.基本知识点 定位: 1.info.plist文件设置 ios8以后,使用定位需要在info.plist文件中添加两个字段NSLocationAlwaysUsageDescription ...

  3. iOS 直播类APP开发流程分解:

    1 . 音视频处理的一般流程: 数据采集→数据编码→数据传输(流媒体服务器) →解码数据→播放显示1.数据采集:摄像机及拾音器收集视频及音频数据,此时得到的为原始数据涉及技术或协议:摄像机:CCD.C ...

  4. windows下与linux下安装redis及redis扩展

    1.        Redis的介绍 Redis是一个开源的使用ANSI C语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value数据库,并提供多种语言的API.从2010年3月15日起 ...

  5. java 调用 C# 类库搞定,三步即可,可以调用任何类及方法,很简单,非常爽啊

    java 调用 C# 类库搞定,三步即可,可以调用任何类及方法,很简单,非常爽啊 java 调用 C# 类库搞定,可以调用任何类及方法,很简单,非常爽啊 总体分三步走: 一.准备一个 C# 类库 (d ...

  6. .NET Mvc Razor也可以这样玩!

    忙碌的工作总是占据了生活的大部分的时间!所以我的博客到现在还是寥寥的几篇文章,技术是用来分享和学习的,对技术有不同的见解,大家都可以分享下,如果如下文章有问题之处请各位指出来,在这个闲下来的时间给大家 ...

  7. Web Essentials之样式表StyleSheets

    返回Web Essentials功能目录 本篇目录 智能感知 视觉提示 验证 Web标准 转换器 Web Essentials中大多数的CSS功能也适用于LESS. 智能感知 生成供应商特定的属性 如 ...

  8. 【腾讯Bugly干货分享】浅谈Android自定义锁屏页的发车姿势

    本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57875330c9da73584b025873 一.为什么需要自定义锁屏页 锁屏 ...

  9. 为自己搭建一个鹊桥 -- Native Page与Web View之间的JSBridge实现方式

    说起JSBridge,大家最熟悉的应该就是微信的WeixinJSBridge,通过它各个公众页面可以调用后台方法和微信进行交互,为用户提供相关功能.我们就来说说UWP下怎么样实现我们自己的JSBrid ...

  10. Java虚拟机12:Java内存模型

    什么是Java内存模型 Java虚拟机规范中试图定义一种Java内存模型(Java Memory Model,JMM)来屏蔽掉各种硬件和操作系统的访问差异,以实现让Java程序在各种平台下都能达到一致 ...