Python-OpenCV学习(八)凸轮廓检测

凸轮廓与Douglas-Peucker算法:
大多数处理轮廓的时候,物体的形状(包括凸形状)都是变换多样的。凸形状内部的任意两点的连线都在该形状里面。
cv.approxPloyDP是一个计算进似多边形框的函数,该函数有三个参数:

  • 第一个参数为“轮廓”
  • 第二个参数为$\varepsilon $值,它表示圆轮廓与近似多边形的最大差值(这个值越小,近似多边形与源轮廓越接近)
  • 第三个参数为“布尔标记”表示这个多边形是否闭和合
    $\varepsilon$是为所得到的近似多边形周长与轮廓之间的最大差值,差值越小,多边形轮廓越相似。

如果需要轮廓的周长信息可以通过Opencv的cv2.arcLength函数来完成:

epsilon=0.01*cv2.arcLength(cnt,True)
approx=cv2.approxPolyDP(cnt,epsilon,True)

通过OpenCV来有效计算一个近似多变形,多边形周长与源轮廓周长之比就为$\varepsilon$
为了计算图形状,需要用OpenCV的cv2.convexHull函数来获取处理过的轮廓信息,

hull =cv2.convexHull(cnt)

完整代码:

import cv2
import numpy as np

img = cv2.pyrDown(cv2.imread("hammer.jpg", cv2.IMREAD_UNCHANGED))

ret, thresh = cv2.threshold(cv2.cvtColor(img.copy(), cv2.COLOR_BGR2GRAY) , 127, 255, cv2.THRESH_BINARY)
black = cv2.cvtColor(np.zeros((img.shape[1], img.shape[0]), dtype=np.uint8), cv2.COLOR_GRAY2BGR)

image, contours, hier = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)

for cnt in contours:
  epsilon = 0.01 * cv2.arcLength(cnt,True)
  approx = cv2.approxPolyDP(cnt,epsilon,True)
  hull = cv2.convexHull(cnt)
  cv2.drawContours(black, [cnt], -1, (0, 255, 0), 2)
  cv2.drawContours(black, [approx], -1, (255, 255, 0), 2)
  cv2.drawContours(black, [hull], -1, (0, 0, 255), 2)

cv2.imshow("hull", black)
cv2.waitKey()
cv2.destroyAllWindows()

1

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要实现手势识别手指的个数,可以使用OpenCV-Python的图像处理和计算机视觉功能。以下是实现手势识别手指的个数的一般步骤: 1. 读取视频或摄像头捕捉的图像。 2. 对图像进行预处理,如去噪、二值化等。 3. 找到手部轮廓。 4. 检测手指。 5. 计算手指的个数。 下面是一个简单的代码示例,可以识别手掌并计算手指的个数: ```python import cv2 import numpy as np # 定义HSV颜色范围 lower_skin = np.array([0, 20, 70], dtype=np.uint8) upper_skin = np.array([20, 255, 255], dtype=np.uint8) cap = cv2.VideoCapture(0) while True: # 读取视频帧 ret, frame = cap.read() # 转换为HSV颜色空间 hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV) # 提取皮肤颜色区域 mask = cv2.inRange(hsv, lower_skin, upper_skin) # 去噪 mask = cv2.erode(mask, None, iterations=2) mask = cv2.dilate(mask, None, iterations=2) # 找到轮廓 contours, hierarchy = cv2.findContours(mask, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE) # 找到最大轮廓 if len(contours) > 0: max_contour = max(contours, key=cv2.contourArea) # 找到包 hull = cv2.convexHull(max_contour, returnPoints=False) # 找到缺陷 defects = cv2.convexityDefects(max_contour, hull) # 计算手指的个数 finger_count = 0 if defects is not None: for i in range(defects.shape[0]): s, e, f, d = defects[i][0] start = tuple(max_contour[s][0]) end = tuple(max_contour[e][0]) far = tuple(max_contour[f][0]) # 计算手指长度 a = np.sqrt((end[0] - start[0])**2 + (end[1] - start[1])**2) b = np.sqrt((far[0] - start[0])**2 + (far[1] - start[1])**2) c = np.sqrt((end[0] - far[0])**2 + (end[1] - far[1])**2) angle = np.arccos((b**2 + c**2 - a**2) / (2*b*c)) # 如果角度小于90度,则为手指 if angle < np.pi/2: finger_count += 1 # 在图像上绘制手指个数 cv2.putText(frame, str(finger_count), (50, 50), cv2.FONT_HERSHEY_SIMPLEX, 1, (0, 255, 0), 2, cv2.LINE_AA) # 显示结果 cv2.imshow('frame', frame) # 按下q键退出 if cv2.waitKey(1) & 0xFF == ord('q'): break # 释放视频捕获对象和销毁所有窗口 cap.release() cv2.destroyAllWindows() ``` 这个代码示例使用肤色检测缺陷检测来识别手指的个数。检测轮廓后,使用`cv2.convexHull`函数找到包,然后使用`cv2.convexityDefects`函数找到缺陷。对于每个缺陷,计算相应的角度,并将其视为手指。最后,在图像上绘制手指个数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值