• OpenCVPython系列之轮廓入门


    本次我们将讨论OpenCV中的轮廓,轮廓属于OpenCV中的一个很重要的部分,同时我们在之前讲过的Canny边缘检测将作为基础。

    轮廓检测

    函数原型:

    cv2.findContours(image, mode, method, contours=None, hierarchy=None, offset=None)

    参数含义:
    image 代表输入的图片。注意输入的图片必须为二值图片。若输入的图片为彩色图片,必须先进行灰度化和二值化。

    mode 表示轮廓的检索模式,有4种:

    cv2.RETR_EXTERNAL  表示只检测外轮廓。

    cv2.RETR_LIST  检测的轮廓不建立等级关系。

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

    cv2.RETR_TREE  建立一个等级树结构的轮廓。

    method  为轮廓的近似办法,有4种:

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

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

    cv2.CHAIN_APPROX_TC89_L1 和 cv2.CHAIN_APPROX_TC89_KCOS使用teh-Chinl chain 近似算法。

    返回值:

    cv2.findContours()函数返回两个值,一个是轮廓本身contours,还有一个是每条轮廓对应的属性hierarchy。

    通常情况下,我们选择参数cv2.CHAIN_APPROX_SIMPLE,因为我们只需要最简单的轮廓点的信息。

    轮廓绘制

    现在我们来看另一个函数:

    cv2.drawContours(image, contours, contourIdx, color, thickness=None, lineType=None, hierarchy=None, maxLevel=None, offset=None)

    第一个参数是指明在哪幅图像上绘制轮廓;image为三通道才能显示轮廓

    第二个参数是轮廓本身,在Python中是一个list;

    第三个参数指定绘制轮廓list中的哪条轮廓,如果是-1,则绘制其中的所有轮廓。后面的参数很简单。其中thickness表明轮廓线的宽度,如果是-1(cv2.FILLED),则为填充模式。

    接下来我们以这幅图为例:

    image.png

    现在我们来进行代码实战:

    import cv2
    import numpy as np
    
    img = cv2.imread("contours.png")
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    contour = cv2.findContours(gray,cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)[0]
    cv2.drawContours(img,contour,-1,(0,0,255),2)
    cv2.imshow("res",img)
    cv2.waitKey(0)
    cv2.destroyAllWindows()

    image.png

    我们还可以改变参数:

    import cv2
    import numpy as np
    
    img = cv2.imread("contours.png")
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    contour = cv2.findContours(gray,cv2.RETR_TREE,cv2.CHAIN_APPROX_SIMPLE)[0]
    cv2.drawContours(img,contour,-1,(0,0,255),2)
    cv2.imshow("res",img)
    cv2.waitKey(0)
    cv2.destroyAllWindows()v

    将RETR_EXTERNAL改为RETR_TREE,改为检测全部轮廓(之前为检测外部轮廓):

    image.png

    现在我们也可以对轮廓进行选取,我们需要修改drawContours的参数,代码:

    import cv2
    import numpy as np
    
    img = cv2.imread("contours.png")
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    contour = cv2.findContours(gray,cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)[0]
    cv2.drawContours(img,contour,1,(0,0,255),2)
    cv2.imshow("res",img)
    cv2.waitKey(0)
    cv2.destroyAllWindows()

    image.png

    通过修改第三个参数我们可以自定义选取哪一个轮廓,这样很方便我们后期的其他操作。

    同样的,我们还可以计算有几个轮廓,适用Numpy,我们来看代码:

    import cv2
    import numpy as np
    
    img = cv2.imread("contours.png")
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    contour = cv2.findContours(gray,cv2.RETR_EXTERNAL,cv2.CHAIN_APPROX_SIMPLE)[0]
    print(np.array(contour).shape)
    cv2.drawContours(img,contour,1,(0,0,255),2)
    cv2.imshow("res",img)
    cv2.waitKey(0)
    cv2.destroyAllWindows()

    image.png

    下面的输出栏会打印出来轮廓的个数,改变参数再次实验一下:

    import cv2
    import numpy as np
    
    img = cv2.imread("contours.png")
    gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
    contour = cv2.findContours(gray,cv2.RETR_TREE,cv2.CHAIN_APPROX_SIMPLE)[0]
    print(np.array(contour).shape)
    cv2.drawContours(img,contour,1,(0,0,255),2)
    cv2.imshow("res",img)
    cv2.waitKey(0)
    cv2.destroyAllWindows()

    image.png

    本次教程对于轮廓进行了一些基本的介绍,下次我们将介绍一些关于轮廓的其他内容。

  • 相关阅读:
    java容器01--初遇
    java虚拟机(1)--运行时数据区
    java虚拟机(2)--垃圾收集
    java虚拟机(3)--内存分配与回收策略
    java虚拟机(4)--类加载机制
    bash编程的信号捕获:
    awk纯干货
    shell中各种括号的作用()、(())、[]、[[]]、{}
    find
    awk
  • 原文地址:https://www.cnblogs.com/wuyuan2011woaini/p/15656423.html
Copyright © 2020-2023  润新知