【菜菜的CV进阶之路-数据预处理-基础】Python轮廓检测、找出轮廓中心点、绘制最小矩形框并裁剪

简介: 【菜菜的CV进阶之路-数据预处理-基础】Python轮廓检测、找出轮廓中心点、绘制最小矩形框并裁剪

一、获取轮廓:

主要使用OpenCV的cv2.findContours()方法:

1. findContours( InputOutputArray image, OutputArrayOfArrays contours,
2.                               OutputArray hierarchy, int mode,
3.                               int method, Point offset=Point());
1. 第一个参数:image,单通道图像矩阵,可以是灰度图,但更常用的是二值图像,一般是经过Canny、拉普拉斯等边缘检测算子处理过的二值图像;
2. 
3. 第二个参数:contours,定义为“vector<vector<Point>> contours”,是一个向量,并且是一个双重向量,向量内每个元素保存了一组由连续的Point点构成的点的集合的向量,每一组Point点集就是一个轮廓。有多少轮廓,向量contours就有多少元素。
4. 
5. 第三个参数:hierarchy,定义为“vector<Vec4i> hierarchy”,先来看一下Vec4i的定义:
6.            typedef    Vec<int, 4>   Vec4i;                                                                                                                                       
7.            Vec4i是Vec<int,4>的别名,定义了一个“向量内每一个元素包含了4个int型变量”的向量。
8. 
9.            所以从定义上看,hierarchy也是一个向量,向量内每个元素保存了一个包含4个int整型的数组。
10. 
11.            向量hiararchy内的元素和轮廓向量contours内的元素是一一对应的,向量的容量相同。
12. 
13.            hierarchy向量内每一个元素的4个int型变量——hierarchy[i][0] ~hierarchy[i][3],分别表示第
14. 
15.         i个轮廓的后一个轮廓、前一个轮廓、父轮廓、内嵌轮廓的索引编号。如果当前轮廓没有对应的后一个
16. 
17.         轮廓、前一个轮廓、父轮廓或内嵌轮廓的话,则hierarchy[i][0] ~hierarchy[i][3]的相应位被设置为
18. 
19.         默认值-1。
20. 
21. 第四个参数:int型的mode,定义轮廓的检索模式:
22.            取值一:CV_RETR_EXTERNAL只检测最外围轮廓,包含在外围轮廓内的内围轮廓被忽略
23.            取值二:CV_RETR_LIST   检测所有的轮廓,包括内围、外围轮廓,但是检测到的轮廓不建立等级关
24.                   系,彼此之间独立,没有等级关系,这就意味着这个检索模式下不存在父轮廓或内嵌轮廓,
25.                   所以hierarchy向量内所有元素的第3、第4个分量都会被置为-1,具体下文会讲到
26.            取值三:CV_RETR_CCOMP  检测所有的轮廓,但所有轮廓只建立两个等级关系,外围为顶层,若外围
27.                   内的内围轮廓还包含了其他的轮廓信息,则内围内的所有轮廓均归属于顶层
28.            取值四:CV_RETR_TREE, 检测所有轮廓,所有轮廓建立一个等级树结构。外层轮廓包含内层轮廓,内
29.                    层轮廓还可以继续包含内嵌轮廓。
30. 
31. 第五个参数:int型的method,定义轮廓的近似方法:
32.            取值一:CV_CHAIN_APPROX_NONE 保存物体边界上所有连续的轮廓点到contours向量内
33.            取值二:CV_CHAIN_APPROX_SIMPLE 仅保存轮廓的拐点信息,把所有轮廓拐点处的点保存入contours
34.                    向量内,拐点与拐点之间直线段上的信息点不予保留
35.            取值三和四:CV_CHAIN_APPROX_TC89_L1,CV_CHAIN_APPROX_TC89_KCOS使用teh-Chinl chain 近
36.                    似算法
37. 第六个参数:Point偏移量,所有的轮廓信息相对于原始图像对应点的偏移量,相当于在每一个检测出的轮廓点上加
38.             上该偏移量,并且Point还可以是负值!
39. 
40.

返回值:image, contours, hierarchy

  • contour返回值
cv2.findContours()函数首先返回一个list,list中每个元素都是图像中的一个轮廓,用numpy中的ndarray表示。
  • hierarchy返回值
1. 该函数还可返回一个可选的hiararchy结果,这是一个ndarray,其中的元素个数和轮廓个数相同,每个轮廓contours[i]对应4个hierarchy元素
2. hierarchy[i][0] ~hierarchy[i][3],分别表示后一个轮廓、前一个轮廓、父轮廓、内嵌轮廓的索引编号,如果没有对应项,则该值为负数。

二、绘制轮廓:

cv2.drawContours()函数

cv2.drawContours(image, contours, contourIdx, color[, thickness[, lineType[, hierarchy[, maxLevel[, offset ]]]]])

  • 第一个参数是指明在哪幅图像上绘制轮廓;
  • 第二个参数是轮廓本身,在Python中是一个list。
  • 第三个参数指定绘制轮廓list中的哪条轮廓,如果是-1,则绘制其中的所有轮廓。后面的参数很简单。其中thickness表明轮廓线的宽度,如果是-1(cv2.FILLED),则为填充模式。绘制参数将在以后独立详细介绍。

为了看到自己画了哪些轮廓,可以使用 cv2.boundingRect()函数获取轮廓的范围,即左上角原点,以及他的高和宽。然后用cv2.rectangle()方法画出矩形轮廓

1. """
2. x, y, w, h = cv2.boundingRect(img)   
3.     参数:
4.     img  是一个二值图
5.     x,y 是矩阵左上点的坐标,
6.     w,h 是矩阵的宽和高
7. 
8. cv2.rectangle(img, (x,y), (x+w,y+h), (0,255,0), 2)
9.     img:       原图
10.     (x,y):   矩阵的左上点坐标
11.     (x+w,y+h):是矩阵的右下点坐标
12.     (0,255,0): 是画线对应的rgb颜色
13.     2:         线宽
14. """
15. for i in range(0,len(contours)):  
16.     x, y, w, h = cv2.boundingRect(contours[i])   
17.     cv2.rectangle(image, (x,y), (x+w,y+h), (153,153,0), 5)

实例:

1. import cv2
2. 
3. img = cv2.imread('wujiaoxing.png')
4. gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
5. ret,binary = cv2.threshold(gray, 127, 255, cv2.THRESH_BINARY)
6. 
7. _,contours, hierarchy = cv2.findContours(binary, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)
8. 
9. draw_img0 = cv2.drawContours(img.copy(),contours,0,(0,255,255),3)
10. draw_img1 = cv2.drawContours(img.copy(),contours,1,(255,0,255),3)
11. draw_img2 = cv2.drawContours(img.copy(),contours,2,(255,255,0),3)
12. draw_img3 = cv2.drawContours(img.copy(), contours, -1, (0, 0, 255), 3)
13. 
14. 
15. print ("contours:类型:",type(contours))
16. print ("第0 个contours:",type(contours[0]))
17. print ("contours 数量:",len(contours))
18. 
19. print ("contours[0]点的个数:",len(contours[0]))
20. print ("contours[1]点的个数:",len(contours[1]))
21. 
22. cv2.imshow("img", img)
23. cv2.imshow("draw_img0", draw_img0)
24. cv2.imshow("draw_img1", draw_img1)
25. cv2.imshow("draw_img2", draw_img2)
26. cv2.imshow("draw_img3", draw_img3)
27. 
28. cv2.waitKey(0)
29. cv2.destroyAllWindows()
30. 
31. 输出:
32. contours:类型: <class 'list'>
33. 第0 个contours: <class 'numpy.ndarray'>
34. contours 数量: 3
35. contours[0]点的个数: 6
36. contours[1]点的个数: 74

三、获取中心点、最小外接矩形:

使用 cv2.minAreaRect(cnt) ,返回点集cnt的最小外接矩形,cnt是所要求最小外接矩形的点集数组或向量,这个点集不定个数。

其中:cnt = np.array([[x1,y1],[x2,y2],[x3,y3],[x4,y4]]) # 必须是array数组的形式

rect = cv2.minAreaRect(cnt) # 得到最小外接矩形的(中心(x,y), (宽,高), 旋转角度)

box = np.int0(cv2.boxPoints(rect)) #通过box会出矩形框

这只是针对某一个轮廓获取最小外接矩形,如果想要获取图像所有轮廓的最小外接矩形则需要将所有轮廓的点集合合并到一个np.array中:

1.     allarea = np.concatenate(contours)#合并所有轮廓点集合
2.     rect = cv2.minAreaRect(allarea)
3.     box = np.int0(cv2.boxPoints(rect))

np.concatenate()函数:

concatenate((a1, a2, …), axis=0)官方文档详解:

1. concatenate(...)
2.     concatenate((a1, a2, ...), axis=0)
3. 
4.     Join a sequence of arrays along an existing axis.
5. 
6.     Parameters
7.     ----------
8.     a1, a2, ... : sequence of array_like
9.         The arrays must have the same shape, except in the dimension
10.         corresponding to `axis` (the first, by default).
11.     axis : int, optional
12.         The axis along which the arrays will be joined.  Default is 0.
13. 
14.     Returns
15.     -------
16.     res : ndarray
17.         The concatenated array.
18. 
19.     See Also
20.     --------
21.     ma.concatenate : Concatenate function that preserves input masks.
22.     array_split : Split an array into multiple sub-arrays of equal or
23.                   near-equal size.
24.     split : Split array into a list of multiple sub-arrays of equal size.
25.     hsplit : Split array into multiple sub-arrays horizontally (column wise)
26.     vsplit : Split array into multiple sub-arrays vertically (row wise)
27.     dsplit : Split array into multiple sub-arrays along the 3rd axis (depth).
28.     stack : Stack a sequence of arrays along a new axis.
29.     hstack : Stack arrays in sequence horizontally (column wise)
30.     vstack : Stack arrays in sequence vertically (row wise)
31.     dstack : Stack arrays in sequence depth wise (along third dimension)

Parameters参数

  • 传入的参数必须是一个多个数组的元组或者列表
  • 另外需要指定拼接的方向,默认是 axis = 0,也就是说对0轴的数组对象进行纵向的拼接(纵向的拼接沿着axis= 1方向);注:一般axis = 0,就是对该轴向的数组进行操作,操作方向是另外一个轴,即axis=1。
1. In [23]: a = np.array([[1, 2], [3, 4]])
2. In [24]: b = np.array([[5, 6]])
3. In [25]: np.concatenate((a, b), axis=0)
4. Out[25]:
5. array([[1, 2],
6.        [3, 4],
7.        [5, 6]])

传入的数组必须具有相同的形状,这里的相同的形状可以满足在拼接方向axis轴上数组间的形状一致即可

如果对数组对象进行 axis= 1 轴的拼接,方向是横向0轴,a是一个2*2维数组,axis= 0轴为2,b是一个1*2维数组,axis= 0 是1,两者的形状不等,这时会报错

1. In [27]: np.concatenate((a,b),axis = 1)
2. ---------------------------------------------------------------------------
3. ValueError                                Traceback (most recent call last)
4. <ipython-input-27-aa1228decc36> in <module>()
5. ----> 1 np.concatenate((a,b),axis = 1)
6. 
7. ValueError: all the input array dimensions except for the concatenation axis must match exactly

将b进行转置,得到b为2*1维数组:

1. In [28]: np.concatenate((a,b.T),axis = 1)
2. Out[28]:
3. array([[1, 2, 5],
4.        [3, 4, 6]])

来源:

[437]numpy库数组拼接np.concatenate()函数_周小董的博客-CSDN博客

findContours函数参数详解_-牧野-的博客-CSDN博客_findcontours函数

OpenCV—Python 轮廓检测 绘出矩形框(findContours\ boundingRect\rectangle)_SongpingWang的博客-CSDN博客_opencv 轮廓分类

python用opencv批量截取图像指定区域_大力挥拳的博客-CSDN博客


AIEarth是一个由众多领域内专家博主共同打造的学术平台,旨在建设一个拥抱智慧未来的学术殿堂!【平台地址:https://devpress.csdn.net/aiearth】 很高兴认识你!加入我们共同进步!

目录
相关文章
|
12天前
|
机器学习/深度学习 监控 TensorFlow
使用Python实现深度学习模型:智能农业病虫害检测与防治
使用Python实现深度学习模型:智能农业病虫害检测与防治
162 65
|
9天前
|
数据采集 人工智能 数据可视化
Python selenium爬虫被检测到,该怎么破?
Python selenium爬虫被检测到,该怎么破?
|
11天前
|
算法 数据可视化 安全
使用Python检测新冠肺炎疫情拐点,抗疫成果明显
使用Python检测新冠肺炎疫情拐点,抗疫成果明显
|
1月前
|
机器学习/深度学习 算法 大数据
【2023年MathorCup高校数学建模挑战赛-大数据竞赛】赛道A:基于计算机视觉的坑洼道路检测和识别 python 代码解析
本文提供了2023年MathorCup高校数学建模挑战赛大数据竞赛赛道A的解决方案,涉及基于计算机视觉的坑洼道路检测和识别任务,包括数据预处理、特征提取、模型建立、训练与评估等步骤的Python代码解析。
43 0
【2023年MathorCup高校数学建模挑战赛-大数据竞赛】赛道A:基于计算机视觉的坑洼道路检测和识别 python 代码解析
|
1月前
|
Python
【Python】实现MATLAB中计算两个矩形相交面积的rectint函数
Python中实现MATLAB中rectint函数的方法,该函数用于计算两个矩形相交区域的面积,并通过定义Rectangle类和calc_area函数展示了如何计算两个矩形的交集面积。
25 1
|
2月前
|
数据采集 网络协议 数据挖掘
网络爬虫进阶之路:深入理解HTTP协议,用Python urllib解锁新技能
【7月更文挑战第30天】网络爬虫是数据分析和信息聚合的关键工具。深入理解HTTP协议及掌握Python的urllib库对于高效爬虫开发至关重要。HTTP协议采用请求/响应模型,具有无状态性、支持多种请求方法和内容协商等特点。
28 3
|
2月前
|
机器学习/深度学习 运维 监控
使用Python实现深度学习模型:智能安防监控与异常检测
【7月更文挑战第26天】 使用Python实现深度学习模型:智能安防监控与异常检测
40 6
|
2月前
|
算法 计算机视觉 Python
|
2月前
|
网络协议 开发者 Python
网络编程小白秒变大咖!Python Socket基础与进阶教程,轻松上手无压力!
【7月更文挑战第25天】在网络技术快速发展的背景下, Python因其简洁的语法和强大的库支持成为学习网络编程的理想选择。
53 5
|
2月前
|
机器学习/深度学习 数据采集 算法
Python编程语言进阶学习:深入探索与高级应用
【7月更文挑战第23天】Python的进阶学习是一个不断探索和实践的过程。通过深入学习高级数据结构、面向对象编程、并发编程、性能优化以及在实际项目中的应用,你将能够更加熟练地运用Python解决复杂问题,并在编程道路上走得更远。记住,理论知识只是基础,真正的成长来自于不断的实践和反思。
下一篇
DDNS