opencv6.5-imgproc图像处理模块之轮廓

2023-12-08 18:32

本文主要是介绍opencv6.5-imgproc图像处理模块之轮廓,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

opencv6.4-imgproc图像处理模块之直方图与模板

这部分的《opencv_tutorial》上都是直接上代码,没有原理部分的解释的。

十一、轮廓

1、图像中找轮廓


  /// 转成灰度并模糊化降噪cvtColor( src, src_gray, CV_BGR2GRAY );blur( src_gray, src_gray, Size(3,3) );
Mat canny_output;//找到轮廓的图vector<vector<Point> > contours;//装载曲线的点vector<Vec4i> hierarchy;/// 用Canny算子检测边缘Canny( src_gray, canny_output, thresh, thresh*2, 3 );/// 寻找轮廓findContours( canny_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) );/// 绘出轮廓Mat drawing = Mat::zeros( canny_output.size(), CV_8UC3 );for( int i = 0; i< contours.size(); i++ )//通过对contours.size(),就知道有几个分离的轮廓了{Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );drawContours( drawing, contours, i, color, 2, 8, hierarchy, 0, Point() );//画出轮廓}/// 在窗体中显示结果namedWindow( "Contours", CV_WINDOW_AUTOSIZE );imshow( "Contours", drawing );
查找轮廓的函数原型:void findContours(InputOutputArray image, OutputArrayOfArrays contours, OutputArray hierarchy, int mode, int method, Point offset=Point())
                                         void findContours(InputOutputArray image, OutputArrayOfArrays contours, int mode, int method, Point offset=Point())

参数列表:image:一个8-bit的单通道原图像。非零的像素值被当作1来处理(类似bool类型)。0的像素值仍旧还是0,所以这个image是被当作二值处理的。你可以使用compare(),inRange(),threshold(),adaptiveThreshold(),Canny(),和其他的从一个灰度或者彩色图中生成一个二值图。该函数会在提出轮廓的时候修改image。如果mode等于CV_RETR_CCOMP或者CV_RETR_FLOODFILL,那么这个输入同样可以是标签的32-bit整数图(CV_32SC1)。

                  contours:检测到的轮廓,每个轮廓都是存储为点向量;

                  hierarchy:可选的输出向量,包含图像拓扑信息。它有着和轮廓数量一样多的元素。对于第i 个轮廓contours[i],元素hierarchy【i】【0】,hiearchy【i】【1】,hiearchy【i】【2】和hiearchy【i】【3】是在contours中基于0索引的,分别表示在同一个层次级别上的下一个轮廓上一个轮廓,第一个孩子轮廓父亲轮廓。如果轮廓 i 没有下一个、上一个、父亲或者嵌套的轮廓,对应的hierarchy【i】的元素就是负的。(这里其实就是个树结构,来进行索引不同的轮廓)

                  mode:轮廓索引模型

                 – CV_RETR_EXTERNAL     只检索最外面的轮廓。它会对所有的轮廓设置 hierarchy[i][2] = hierarchy[i][3] = -1 .
                 – CV_RETR_LIST                 不建立任何层次关系来进行索引所有的轮廓.
                 – CV_RETR_CCOMP       索引所有的轮廓并将它们组织成一个two-level的hierarchy(两个层次的层级关系):在顶层上,有着成分的外部边界;在第二层是孔洞的边界。如果有另一个轮廓在一个连接起来的成分内部,那么就将它放在顶层上。
                – CV_RETR_TREE        检索所有的轮廓并重构一个全层次的嵌套轮廓结构,在contours.c的例子中你更可以看到全层次建立的代码。
                 method :轮廓逼近的方法

                   – CV_CHAIN_APPROX_NONE   存储所有轮廓点的绝对值。也就是说任何的轮廓的2子序列点 (x1,y1) 和 (x2,y2) 就是表示水平的,竖直的或者对角线邻居,也就是: max(abs(x1-x2),abs(y2-y1))==1;
                – CV_CHAIN_APPROX_SIMPLE    压缩水平的,竖直的和对角线线段,只保留他们的终端。例如:对于一个up-right 的矩形轮廓是由4个点进行编码的。

                 – CV_CHAIN_APPROX_TC89_L1,  CV_CHAIN_APPROX_TC89_KCOS     使用 Teh-Chin 链逼近算法中主流之一的算法。

                offset:可选的偏移量,通过这个值可以平移每一个轮廓。当轮廓是从一个图像的ROI中提取而你需要在整个图像上下文中分析的时候会变得很有用。

该函数使用算法从二值图像中索引轮廓。轮廓对于形状分析和对象的检测识别是很有用的,在squares.c中有例子代码。

Notes:image会被该函数所修改,同样的该函数不考虑图像的1像素边界(它会被0填充然后用来作为邻居分析),因此接触图像边界的轮廓会被修剪(clip,夹)。

画出轮廓的函数原型:void drawContours(InputOutputArray image, InputArrayOfArrays contours, int contourIdx, const Scalar& color, int thickness=1, int lineType=LINE_8, InputArray hierarchy= noArray(), int maxLevel=INT_MAX, Point offset=Point() );

参数列表:目标图像、输入的轮廓、轮廓的索引id、轮廓的颜色、粗度、线类型、可选的层次信息、最大级别、偏移量;

   第一个参数:目标图像,即画布;

   第二个参数:所有的输入轮廓,每个轮廓存储成点向量的形式;

   第三个参数:用来索引需要画的轮廓的参数,如果是负的,那么就画所有的轮廓;

   第四个参数:轮廓的颜色;

   第五个参数:画的轮廓的线的粗细程度,如果是负的(例如:thickness = CV_FILLED),轮廓内部也会画出来;

   第六个参数:线连接类型,line()可以有更详细的说明。

   第七个参数:可选的关于层次的信息。当你只想画某些轮廓的时候才是需要的(见maxLevel);

   第八个参数:画轮廓的最大等级。如果为0,那么只画指定的轮廓。如果为1,该函数会画轮廓及所有的嵌套轮廓。如果为2,该函数会画轮廓、所有的嵌套轮廓、所有的嵌套-to-嵌套的轮廓,以此类推。该参数当hierarchy可用的时候才被考虑;

   第九个参数:可选的轮廓平移参数。通过制定的平移量offset = (dx,dy)来平移所有的画的轮廓。

该函数但thickness >=0的时候会画轮廓的外部线条,或者当thickness <0的时候会填充轮廓的框起来的区域。下面的代码是展示如何从二值图像中索引联通的成分,并且标记他们:

Mat src;src = imread(name,0);
Mat dst = Mat::zeros(src.rows, src.cols, CV_8UC3);src = src > 1;vector<vector<Point> > contours;
vector<Vec4i> hierarchy;
findContours( src, contours, hierarchy,RETR_CCOMP, CHAIN_APPROX_SIMPLE );
//通过对top-level的轮廓进行迭代来画出每个联通的成分,使用随机颜色int idx = 0;for( ; idx >= 0; idx = hierarchy[idx][0] ){Scalar color( rand()&255, rand()&255, rand()&255 );drawContours( dst, contours, idx, color, FILLED, 8, hierarchy );
}namedWindow( "Components", 1 );imshow( "Components", dst );waitKey(0);
2、计算物体的凸包

凸包就是将对象进行外部轮廓的包含,而且是凸图形的:


/// 转成灰度图并进行模糊降噪cvtColor( src, src_gray, CV_BGR2GRAY );blur( src_gray, src_gray, Size(3,3) );
Mat src_copy = src.clone();Mat threshold_output;vector<vector<Point> > contours;//存储轮廓的点集合vector<Vec4i> hierarchy;//构建轮廓的层次结构/// 对图像进行二值化
int thresh = 100;int max_thresh = 255;RNG rng(12345);
threshold( src_gray, threshold_output, thresh, 255, THRESH_BINARY ); /// 寻找轮廓 findContours( threshold_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) ); /// 对每个轮廓计算其凸包 vector<vector<Point> >hull( contours.size() ); for( int i = 0; i < contours.size(); i++ ) {  
     convexHull( Mat(contours[i]), hull[i], false );//凸包计算
     }/// 绘出轮廓及其凸包Mat drawing = Mat::zeros( threshold_output.size(), CV_8UC3 );for( int i = 0; i< contours.size(); i++ ){Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );drawContours( drawing, contours, i, color, 1, 8, vector<Vec4i>(), 0, Point() );//画轮廓drawContours( drawing, hull, i, color, 1, 8, vector<Vec4i>(), 0, Point() );//画凸包}/// 把结果显示在窗体namedWindow( "Hull demo", CV_WINDOW_AUTOSIZE );imshow( "Hull demo", drawing );
凸包函数原型:void convexHull(InputArray points, OutputArray hull, bool clockwise=false, bool returnPoints=true);

参数列表:2d点集合、输出的凸包、定向标识、操作标识;

第一个参数:输入的2D点集合,存储在vector或者Mat中。

第二个参数:输出的凸包,是一个包含索引的整数向量或者是点向量。在前者中,这个hull元素是在原始数组中的凸包点上基于0索引的;后者中,hull元素自身就是凸包点。(也就是说该参数要不是索引到原始图像找凸包点,或者是直接提取凸包点);

第三个参数:定向标识,如果为true,那么输出的凸包就是顺时针方向的;不然就是逆时针方向的。假设的坐标系的x 轴指向右边,y 轴指向上面;

第四个参数:操作标识,在矩阵Mat的情况中,当这个参数为true,该函数返回凸包点;否则返回指向凸包点的索引值;当在第二个参数是数组vector的情况中,该标识被忽略,输出是依赖vector的类型指定的,vector<int> 暗示returnPoints= true ,vector<Point>暗示 returnPoints= false。

    该函数使用Sklansky的算法来查找一个2d点集的凸包,在当前的执行情况下的复杂度是O(NlogN),可以参见convexhull.cpp中验证不同的函数变量的结果。

3、给轮廓加上矩形或者圆形边界框


这个还是挺有用的,有时候识别一个物体,想先用简单的数字图像处理方法得到更少的区域然后可以提取从而接着使用模式识别的方法进行训练和建立分类器。

使用OpenCV函数 boundingRect 来计算包围轮廓的矩形框.

使用OpenCV函数 minEnclosingCircle 来计算完全包围已有轮廓最小圆.

int thresh = 100;
int max_thresh = 255;
RNG rng(12345);
/// 转化成灰度图像并进行平滑 用来减少噪音点cvtColor( src, src_gray, CV_BGR2GRAY );blur( src_gray, src_gray, Size(3,3) );
Mat threshold_output;vector<vector<Point> > contours;//存储轮廓点vector<Vec4i> hierarchy;//构建不同轮廓的层次结构/// 使用Threshold检测边缘threshold( src_gray, threshold_output, thresh, 255, THRESH_BINARY );/// 找到轮廓findContours( threshold_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) );/// 多边形逼近轮廓 + 获取矩形和圆形边界框vector<vector<Point> > contours_poly( contours.size() );//创建contours.size()个空的多边形vector<Rect> boundRect( contours.size() );//创建contours.size()个矩形框vector<Point2f>center( contours.size() );//创建contours.size()个圆心
  vector<float>radius( contours.size() );//创建contours.size()个半径for( int i = 0; i < contours.size(); i++ ){ 
       approxPolyDP( Mat(contours[i]), contours_poly[i], 3, true );//多边形逼近boundRect[i] = boundingRect( Mat(contours_poly[i]) );//获取某个轮廓的矩形框minEnclosingCircle( contours_poly[i], center[i], radius[i] );//生成某个轮廓的最小包含圆的圆心和半径}/// 画多边形轮廓 + 包围的矩形框 + 圆形框Mat drawing = Mat::zeros( threshold_output.size(), CV_8UC3 );for( int i = 0; i< contours.size(); i++ ){Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );//随机颜色drawContours( drawing, contours_poly, i, color, 1, 8, vector<Vec4i>(), 0, Point() );//画轮廓-多边形rectangle( drawing, boundRect[i].tl(), boundRect[i].br(), color, 2, 8, 0 );//画矩形circle( drawing, center[i], (int)radius[i], color, 2, 8, 0 );//画圆形}/// 显示在一个窗口namedWindow( "Contours", CV_WINDOW_AUTOSIZE );imshow( "Contours", drawing );
多边形逼近的函数原型:void approxPolyDP(InputArray curve, OutputArray approxCurve, double epsilon, bool closed);

参数列表:输入数组、逼近的结果、逼近的精度、是否闭合的标识;

第一个参数:一个2d点的输入向量可以存储在:

                          – std::vector or Mat    (C++ interface)
                         – Nx2 numpy array      (Python interface)
                        – CvSeq or ‘‘ CvMat      (C interface)

第二个参数:逼近的结果,该参数的类型应该匹配输入曲线的类型。(在c接口中,逼近的曲线存储在内存存储区中,返回的是指向该内存的指针,我列出来的都是cpp接口,所以该句可忽略);

第三个参数:指定逼近精度的参数。这是介于原始曲线与它的逼近之间的最大的距离;

第四个参数:如果为真,逼近的曲线是闭合的(即,将首尾连起来);否则就不是闭合的。

该函数使用一个更少顶点的曲线/多边形来逼近另一个曲线或多边形,使得介于他们之间的距离小于或者等于指定的精度。该函数使用的是Douglas-Peucker算法。http://en.wikipedia.org/wiki/Ramer-Douglas-Peucker_algorithm


计算一个点集合的up-right边界矩形的函数原型:Rect boundingRect(InputArray points);

参数列表:输入点;

第一个参数:输入的2D点集合,存储在vector或者Mat中。

该函数对指定的点集合计算并返回最小的up-right矩形。


计算一个2d点集合的最小区域圆的函数原型:void minEnclosingCircle(InputArray points, Point2f& center, float& radius)

参数列表:输入的点集合、输出的圆心坐标、输出的半径;

第一个参数:2D点的输入向量,存储在:

                        – std::vector<> or Mat (C++ interface)
                         – CvSeq* or CvMat* (C interface)
                        – Nx2 numpy array (Python interface)

第二个参数:圆的输出的圆心;

第三个参数:圆的输出的半径

该函数在一个2D点集合中使用一个迭代算法来查找最接近的圆,见minarea.cpp中有例子。

4、给轮廓加上倾斜的边界框和椭圆

该部分与上面部分很相似,不过采用的函数是不同的。


int thresh = 100;
int max_thresh = 255;
RNG rng(12345);
 /// 转为灰度图并模糊化 来减小噪音点cvtColor( src, src_gray, CV_BGR2GRAY );blur( src_gray, src_gray, Size(3,3) );
 Mat threshold_output;vector<vector<Point> > contours;//存储轮廓的向量vector<Vec4i> hierarchy;//轮廓的层次结构/// 阈值化检测边界threshold( src_gray, threshold_output, thresh, 255, THRESH_BINARY );/// 寻找轮廓findContours( threshold_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) );/// 对每个找到的轮廓创建可倾斜的边界框和椭圆vector<RotatedRect> minRect( contours.size() );//存储contours.size()个旋转的边界框***************此处是重点,RotateRectvector<RotatedRect> minEllipse( contours.size() );存储contours.size()个旋转的椭圆*************for( int i = 0; i < contours.size(); i++ ){ 
        minRect[i] = minAreaRect( Mat(contours[i]) );//获取包含最小区域的矩形if( contours[i].size() > 5 ){ 
         minEllipse[i] = fitEllipse( Mat(contours[i]) );//获取椭圆所需的信息
         }}/// 绘出轮廓及其可倾斜的边界框和边界椭圆Mat drawing = Mat::zeros( threshold_output.size(), CV_8UC3 );for( int i = 0; i< contours.size(); i++ ){Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );// contourdrawContours( drawing, contours, i, color, 1, 8, vector<Vec4i>(), 0, Point() );//画对象的轮廓// ellipseellipse( drawing, minEllipse[i], color, 2, 8 );//画椭圆// rotated rectanglePoint2f rect_points[4]; 
       minRect[i].points( rect_points );//将旋转矩形的顶点赋值给实参;for( int j = 0; j < 4; j++ )line( drawing, rect_points[j], rect_points[(j+1)%4], color, 1, 8 );//画出倾斜矩形,采用的是画四条线的形式实现的}/// 结果在窗体中显示namedWindow( "Contours", CV_WINDOW_AUTOSIZE );imshow( "Contours", drawing );
在输入的2D点集合中找到一个最小区域的旋转后的矩形的函数原型:RotatedRect minAreaRect(InputArray points)
参数列表:输入的2D点集合
第一个参数:输入的2D点的向量,存储在:
                               – std::vector<> or Mat (C++ interface)

                              – CvSeq* or CvMat* (C interface)
                              – Nx2 numpy array (Python interface)

该函数计算并返回一个指定的点集合的最小区域边界的矩形,在minarea.cpp中有例子,开发者应该注意当数据是接近包含的矩阵元素边界的时候 返回的旋转矩形可以包含负指数。


沿着一个2D点集合进行拟合一个椭圆的函数原型:RotatedRect fitEllipse(InputArray points)

第一个参数:输入的2D点集合,可以存储在:

                      – std::vector<> or Mat (C++ interface)
                     – CvSeq* or CvMat* (C interface)
                     – Nx2 numpy array (Python interface)

该函数计算在对一个2D点集合拟合的最好的椭圆(最小二乘作为损失函数来判别)。它返回旋转的矩形中内含的椭圆,使用的算法是[FItzgibbon95].开发者应该注意当数据点很靠近包含的矩阵元素的边界的时候,返回的椭圆/旋转矩形数据包含负指数例子代码在fitellipse.cpp中。

5、计算轮廓的矩

使用OpenCV函数 moments 计算图像所有的矩(最高到3阶)

使用OpenCV函数 contourArea 来计算轮廓面积

使用OpenCV函数 arcLength 来计算轮廓或曲线长度

int thresh = 100;
int max_thresh = 255;
RNG rng(12345);
 /// 把原图像转化成灰度图像并进行平滑cvtColor( src, src_gray, CV_BGR2GRAY );blur( src_gray, src_gray, Size(3,3) );
Mat canny_output;vector<vector<Point> > contours;//存储轮廓vector<Vec4i> hierarchy;//存储轮廓的层次结构/// 使用Canndy检测边缘Canny( src_gray, canny_output, thresh, thresh*2, 3 );/// 找到轮廓findContours( canny_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) );/// 计算矩vector<Moments> mu(contours.size() );for( int i = 0; i < contours.size(); i++ ){ 
         mu[i] = moments( contours[i], false );//计算轮廓的矩
      }///  计算中心矩:vector<Point2f> mc( contours.size() );for( int i = 0; i < contours.size(); i++ ){ 
          mc[i] = Point2f( mu[i].m10/mu[i].m00 , mu[i].m01/mu[i].m00 ); 
      }/// 绘制轮廓Mat drawing = Mat::zeros( canny_output.size(), CV_8UC3 );for( int i = 0; i< contours.size(); i++ ){Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );drawContours( drawing, contours, i, color, 2, 8, hierarchy, 0, Point() );circle( drawing, mc[i], 4, color, -1, 8, 0 );}/// 显示到窗口中namedWindow( "Contours", CV_WINDOW_AUTOSIZE );imshow( "Contours", drawing );/// 通过m00计算轮廓面积并且和OpenCV函数比较printf("\t Info: Area and Contour Length \n");for( int i = 0; i< contours.size(); i++ ){printf(" * Contour[%d] - Area (M_00) = %.2f - Area OpenCV: %.2f - Length: %.2f \n", 
              i, mu[i].m00, contourArea(contours[i]), arcLength( contours[i], true ) //第几个轮廓,轮廓的空间矩,轮廓的面积,轮廓的周长
               );//计算轮廓面积和 轮廓或曲线的长度Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );drawContours( drawing, contours, i, color, 2, 8, hierarchy, 0, Point() );circle( drawing, mc[i], 4, color, -1, 8, 0 );

计算一个多边形或者光栅形状所有的矩最高到3阶的函数原型:Moments moments(InputArray array, bool binaryImage=false )

参数列表:输入数组、标识

第一个参数:光栅图像(单通道,8-bit或者浮点2D数组);或者一个2D点(Point 或者 Point2f)的数组(1xN或者Nx1);

第二个参数:如果为真,所有的非0图像像素被视为 1 ;该参数只用在图像上。

该函数计算一个向量形状或者光栅形状的矩到3阶。结果返回在结构体Moments。

class Moments
{
public:
<span style="white-space:pre">		</span>Moments();
<span style="white-space:pre">		</span>Moments(double m00, double m10, double m01, double m20, double m11,
<span style="white-space:pre">		</span>double m02, double m30, double m21, double m12, double m03 );
<span style="white-space:pre">		</span>Moments( const CvMoments& moments );
<span style="white-space:pre">		</span>operator CvMoments() const;
// spatial moments
<span style="white-space:pre">		</span>double m00, m10, m01, m20, m11, m02, m30, m21, m12, m03;
// central moments
<span style="white-space:pre">		</span>double mu20, mu11, mu02, mu30, mu21, mu12, mu03;
// central normalized moments
<span style="white-space:pre">		</span>double nu20, nu11, nu02, nu30, nu21, nu12, nu03;
}
在光栅图像的时候,空间矩 以下面的方式计算:

 

中心矩以下面的方式计算:

这里是团块中心:

归一化的中心矩按照下面的方式计算:

notes:
轮廓的矩以同样的方式定义,不过是使用Green(http://en.wikipedia.org/wiki/Green_theorem).)的公式计算的。所以因为一个受限的光栅分辨率,轮廓的矩的计算轻微的不同于同样的光栅轮廓的矩计算。

因为轮廓矩是使用Green公式计算的,所以可能看到奇怪的与自相交的轮廓结果,例如:一个蝴蝶形状轮廓的0区域(m00)。


计算一个轮廓的面积的函数原型:double contourArea(InputArray contour, bool oriented=false )

第一个参数:输入的2D点的向量(轮廓索引),存储在vector或者Mat中;

第二个参数:定向区域标识。如果为true,该函数返回一个有符号的面积值,依赖于轮廓的方向(顺时针或者逆时针)。使用这个特征,你可以通过使用一个区域的符号决定一个轮廓的方向。默认情况下,该参数是false的,这意味着会返回绝对值。

该函数计算一个轮廓的面积,相似于moment()。该面积是使用Green公式计算的。所以,如果你通过使用drawContours()或者fillPoly()画这个轮廓,返回的面积和非0像素的个数,会不一样。同样的该函数差不多会在一个有着自相交的轮廓上给出一个错误的结果。(也就是这时候该函数不靠谱)。

例子代码:

   vector<Point> contour;
contour.push_back(Point2f(0, 0));
contour.push_back(Point2f(10, 0));
contour.push_back(Point2f(10, 10));
contour.push_back(Point2f(5, 4));double area0 = contourArea(contour);
vector<Point> approx;approxPolyDP(contour, approx, 5, true);double area1 = contourArea(approx);
cout << "area0 =" << area0 << endl <<
<span style="white-space:pre">	</span>"area1 =" << area1 << endl <<
<span style="white-space:pre">	</span>"approx poly vertices" << approx.size() << endl;

计算一个轮廓的周边或者一个曲线长度的函数原型:double arcLength(InputArray curve, bool closed)

参数列表:

第一个参数:输入的2D点向量,存储在vector或者Mat中;

第二个参数:表示该曲线是否闭合。

该函数计算一个曲线的长度或者一个闭合轮廓的周长。

6、多边形测试


是执行一个轮廓中点的测试。

 /// 创建一个图形     const int r = 100;Mat src = Mat::zeros( Size( 4*r, 4*r ), CV_8UC1 );//创建画布/// 绘制一系列点创建一个轮廓:vector<Point2f> vert(6);//一个有6个点的轮廓vert[0] = Point( 1.5*r, 1.34*r );vert[1] = Point( 1*r, 2*r );vert[2] = Point( 1.5*r, 2.866*r );vert[3] = Point( 2.5*r, 2.866*r );vert[4] = Point( 3*r, 2*r );vert[5] = Point( 2.5*r, 1.34*r );//自己定轮廓的6各点/// 在src内部绘制for( int j = 0; j < 6; j++ ){ 
      line( src, vert[j],  vert[(j+1)%6], Scalar( 255 ), 3, 8 ); //将该轮廓画出来
      }/// 得到轮廓vector<vector<Point> > contours; vector<Vec4i> hierarchy;Mat src_copy = src.clone();findContours( src_copy, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE);//查找轮廓/// 计算到轮廓的距离Mat raw_dist( src.size(), CV_32FC1 );for( int j = 0; j < src.rows; j++ ){ 
    for( int i = 0; i < src.cols; i++ ){ 
           raw_dist.at<float>(j,i) = pointPolygonTest( contours[0], Point2f(i,j), true ); 整个画布,逐点测试到第0个轮廓的距离
          }}double minVal; double maxVal;minMaxLoc( raw_dist, &minVal, &maxVal, 0, 0, Mat() );//查找全局最大最小minVal = abs(minVal); maxVal = abs(maxVal);/// 图形化的显示距离Mat drawing = Mat::zeros( src.size(), CV_8UC3 );for( int j = 0; j < src.rows; j++ ){ 
      for( int i = 0; i < src.cols; i++ ) {if( raw_dist.at<float>(j,i) < 0 ){ 
               drawing.at<Vec3b>(j,i)[0] = 255 - (int) abs(raw_dist.at<float>(j,i))*255/minVal;
             }else if( raw_dist.at<float>(j,i) > 0 ){
               drawing.at<Vec3b>(j,i)[2] = 255 - (int) raw_dist.at<float>(j,i)*255/maxVal;
              }else{ 
              drawing.at<Vec3b>(j,i)[0] = 255; drawing.at<Vec3b>(j,i)[1] = 255; drawing.at<Vec3b>(j,i)[2] = 255;
              }}}/// 创建窗口显示结果char* source_window = "Source";namedWindow( source_window, CV_WINDOW_AUTOSIZE );imshow( source_window, src );namedWindow( "Distance", CV_WINDOW_AUTOSIZE );imshow( "Distance", drawing );
执行一个轮廓中点的测试的函数原型:double pointPolygonTest(InputArray contour, Point2f pt, bool measureDist)
参数列表:输入的轮廓、测试的点、标识;

第一个参数:输入的轮廓;

第二个参数:针对某个的测试的点;

第三个参数:如果为真,该函数将评估一个点到最近的轮廓的边界的有符号距离;否则只检查该点是否在轮廓内部。

该函数决策着该点在一个轮廓的内部,外部,或者刚好在轮廓的边界上(或者刚好在轮廓的顶点上)。该函数返回为正(内部),负(外部),0(在边界上)。当measureDist = false,该函数返回+1,-1,0;不然返回的值是一个点到最近轮廓边的有符号距离值。


图像处理部分的例子算是都过了一遍了,接下来是ml部分

这篇关于opencv6.5-imgproc图像处理模块之轮廓的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/470915

相关文章

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit

Python模块导入的几种方法实现

《Python模块导入的几种方法实现》本文主要介绍了Python模块导入的几种方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录一、什么是模块?二、模块导入的基本方法1. 使用import整个模块2.使用from ... i

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

Jenkins构建Maven聚合工程,指定构建子模块

一、设置单独编译构建子模块 配置: 1、Root POM指向父pom.xml 2、Goals and options指定构建模块的参数: mvn -pl project1/project1-son -am clean package 单独构建project1-son项目以及它所依赖的其它项目。 说明: mvn clean package -pl 父级模块名/子模块名 -am参数

寻迹模块TCRT5000的应用原理和功能实现(基于STM32)

目录 概述 1 认识TCRT5000 1.1 模块介绍 1.2 电气特性 2 系统应用 2.1 系统架构 2.2 STM32Cube创建工程 3 功能实现 3.1 代码实现 3.2 源代码文件 4 功能测试 4.1 检测黑线状态 4.2 未检测黑线状态 概述 本文主要介绍TCRT5000模块的使用原理,包括该模块的硬件实现方式,电路实现原理,还使用STM32类

python内置模块datetime.time类详细介绍

​​​​​​​Python的datetime模块是一个强大的日期和时间处理库,它提供了多个类来处理日期和时间。主要包括几个功能类datetime.date、datetime.time、datetime.datetime、datetime.timedelta,datetime.timezone等。 ----------动动小手,非常感谢各位的点赞收藏和关注。----------- 使用datet

C8T6超绝模块--EXTI

C8T6超绝模块–EXTI 大纲 控制流程结构体分析EXTI实现按键 具体案例 控制流程 这里是流程框图,具体可以去看我STM32专栏的EXTI的具体分析 结构体分析 typedef struct {uint32_t EXTI_Line; // 中断/事件线EXTIMode_TypeDef EXTI_Mode; // EXTI 模式EXTITrigger_TypeDef EXTI_