【转载】OpenCV-Python系列之轮廓入门(二十七)

本次我们将讨论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),则为填充模式。

接下来我们以这幅图为例:
在这里插入图片描述

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

view plaincopy to clipboardprint?
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()  

在这里插入图片描述

我们还可以改变参数:

view plaincopy to clipboardprint?
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()  

将RETR_EXTERNAL改为RETR_TREE,改为检测全部轮廓(之前为检测外部轮廓):
在这里插入图片描述

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

view plaincopy to clipboardprint?
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()  

在这里插入图片描述

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

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

view plaincopy to clipboardprint?
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()  

在这里插入图片描述

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

view plaincopy to clipboardprint?
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()  

在这里插入图片描述

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

查看文章汇总页https://blog.csdn.net/weixin_44237705/article/details/107864965
更多openvino技术信息可以入群交流~
在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/weixin_44237705/article/details/108572914