位置: IT常识 - 正文

【计算机视觉】基于Python—OpenCV的手势识别详解(一)(计算机视觉算法)

编辑:rootadmin
【计算机视觉】基于Python—OpenCV的手势识别详解(一) 文章目录更新日记前言前期准备识别手部模型识别视频输入方法手势识别方法完整代码结语更新日记

推荐整理分享【计算机视觉】基于Python—OpenCV的手势识别详解(一)(计算机视觉算法),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:计算机视觉技术的应用,计算机视觉技术,计算机视觉与图像处理,计算机视觉与图像处理,计算机视觉的未来发展方向有哪些,计算机视觉的未来发展方向有哪些,计算机视觉技术,计算机视觉技术,内容如对您有帮助,希望把文章链接给更多的朋友!

更新日记: 2022.04.18:应各位网友需求,已mp库更新后的手部识别模型代码。目前可以正常的RUN啦!!

前言

人工智能的浪潮正在席卷全球。一个已经被谈论了几十年的概念,如今这几年,相关技术的发展速度越来越快。机器学习、深度学习、计算机视觉等名词逐渐走进人们的生活,它们同属于人工智能的范畴之中。

计算机视觉是人工智能领域的一个分支计算机视觉实际上是一个跨领域的交叉学科,包括计算机科学,数学,工程学,物理学,生物学和心理学等领域。许多科学家认为,计算机视觉为人工智能的发展开拓了道路。

简单来说,计算机视觉就是赋予计算机一双观察世界的眼睛,再使用计算机优秀的大脑快速的计算,服务人类。

今天我们将深入浅出,简单介绍Python计算机视觉中的手势识别方法,识别手势——数字(一、二、三、四、五和大拇指的赞赏)。如果你喜欢本篇文章或对你有帮助的话,别忘了点赞+关注噢!

前期准备

本篇我们将会用到Python的OpenCV模块和手部模型模块mediapipe,在Python的pip安装方法中,安装方法如下:

opencv是常用的图像识别模块

mediapipe是谷歌开发并开源的多媒体机器学习模型应用框架。

pip install opencv-pythonpip install mediapipe

如果你的电脑装有Anaconda,建议选择在Anaconda的环境命令行中进行相应模块的安装,以此构建更具体的机器学习环境

当你安装好OpenCV和mediapipe模块以后,你可以在Python代码中写入

import cv2import mediapipe as mp

如果运行成功,那么你的Opencv-python模块即为安装成功,那么我们现在就开始进入今天的正题吧!

识别手部模型

既然要做手势识别,那么就要去找到我们传入图像的手部信息。本处我们将使用mediapipe模型去找到手部模型,并完成手部模型的识别模块,并命名,我们将在后续手势识别内容中将其作为模块引入

HandTrackingModule.py

# -*- coding:utf-8 -*-"""CODE >>> SINCE IN CAIXYPROMISE.MOTTO >>> STRIVE FOR EXCELLENT.CONSTANTLY STRIVING FOR SELF-IMPROVEMENT.@ By: CaixyPromise@ Date: 2021-10-17"""import cv2import mediapipe as mpclass HandDetector: """ 使用mediapipe库查找手。导出地标像素格式。添加了额外的功能。 如查找方式,许多手指向上或两个手指之间的距离。而且提供找到的手的边界框信息。 """ def __init__(self, mode=False, maxHands=2, detectionCon=0.5, minTrackCon = 0.5): """ :param mode: 在静态模式下,对每个图像进行检测 :param maxHands: 要检测的最大手数 :param detectionCon: 最小检测置信度 :param minTrackCon: 最小跟踪置信度 """ self.mode = mode self.maxHands = maxHands self.modelComplex = False self.detectionCon = detectionCon self.minTrackCon = minTrackCon# 初始化手部识别模型 self.mpHands = mp.solutions.hands self.hands = self.mpHands.Hands(self.mode, self.maxHands, self.modelComplex, self.detectionCon, self.minTrackCon) self.mpDraw = mp.solutions.drawing_utils# 初始化绘图器 self.tipIds = [4, 8, 12, 16, 20]# 指尖列表 self.fingers = [] self.lmList = [] def findHands(self, img, draw=True): """ 从图像(BRG)中找到手部。 :param img: 用于查找手的图像。 :param draw: 在图像上绘制输出的标志。 :return: 带或不带图形的图像 """ imgRGB = cv2.cvtColor(img, cv2.COLOR_BGR2RGB) # 将传入的图像由BGR模式转标准的Opencv模式——RGB模式, self.results = self.hands.process(imgRGB) if self.results.multi_hand_landmarks: for handLms in self.results.multi_hand_landmarks: if draw: self.mpDraw.draw_landmarks(img, handLms, self.mpHands.HAND_CONNECTIONS) return img def findPosition(self, img, handNo=0, draw=True): """ 查找单手的地标并将其放入列表中像素格式。还可以返回手部周围的边界框。 :param img: 要查找的主图像 :param handNo: 如果检测到多只手,则为手部id :param draw: 在图像上绘制输出的标志。(默认绘制矩形框) :return: 像素格式的手部关节位置列表;手部边界框 """ xList = [] yList = [] bbox = [] bboxInfo =[] self.lmList = [] if self.results.multi_hand_landmarks: myHand = self.results.multi_hand_landmarks[handNo] for id, lm in enumerate(myHand.landmark): h, w, c = img.shape px, py = int(lm.x * w), int(lm.y * h) xList.append(px) yList.append(py) self.lmList.append([px, py]) if draw: cv2.circle(img, (px, py), 5, (255, 0, 255), cv2.FILLED) xmin, xmax = min(xList), max(xList) ymin, ymax = min(yList), max(yList) boxW, boxH = xmax - xmin, ymax - ymin bbox = xmin, ymin, boxW, boxH cx, cy = bbox[0] + (bbox[2] // 2), \ bbox[1] + (bbox[3] // 2) bboxInfo = {"id": id, "bbox": bbox,"center": (cx, cy)} if draw: cv2.rectangle(img, (bbox[0] - 20, bbox[1] - 20), (bbox[0] + bbox[2] + 20, bbox[1] + bbox[3] + 20), (0, 255, 0), 2) return self.lmList, bboxInfo def fingersUp(self): """ 查找列表中打开并返回的手指数。会分别考虑左手和右手 :return:竖起手指的列表 """ if self.results.multi_hand_landmarks: myHandType = self.handType() fingers = [] # Thumb if myHandType == "Right": if self.lmList[self.tipIds[0]][0] > self.lmList[self.tipIds[0] - 1][0]: fingers.append(1) else: fingers.append(0) else: if self.lmList[self.tipIds[0]][0] < self.lmList[self.tipIds[0] - 1][0]: fingers.append(1) else: fingers.append(0) # 4 Fingers for id in range(1, 5): if self.lmList[self.tipIds[id]][1] < self.lmList[self.tipIds[id] - 2][1]: fingers.append(1) else: fingers.append(0) return fingers def handType(self): """ 检查传入的手部是左还是右 :return: "Right" 或 "Left" """ if self.results.multi_hand_landmarks: if self.lmList[17][0] < self.lmList[5][0]: return "Right" else: return "Left"识别视频输入方法【计算机视觉】基于Python—OpenCV的手势识别详解(一)(计算机视觉算法)

完成手部模型的获取与识别,现在我们就要将内容传入到计算机当中,使其能进行手部的识别以及手势的识别。本处我们将使用OpenCV进行内容的输入流,开启计算机的摄像头获取内容,并使用刚刚我们写的HandTrackingModule模块作为手部的识别模块。

Main.py

# -*- coding:utf-8 -*-"""CODE >>> SINCE IN CAIXYPROMISE.MOTTO >>> STRIVE FOR EXCELLENT.CONSTANTLY STRIVING FOR SELF-IMPROVEMENT.@ By: CaixyPromise@ Date: 2021-10-17"""import cv2from HandTrackingModule import HandDetectorclass Main: def __init__(self): self.camera = cv2.VideoCapture(0,cv2.CAP_DSHOW) # 以视频流传入 self.camera.set(3, 1280) # 设置分辨率 self.camera.set(4, 720) def Gesture_recognition(self): while True: self.detector = HandDetector() frame, img = self.camera.read() img = self.detector.findHands(img) # 找到你的手部 lmList, bbox = self.detector.findPosition(img) # 获取你手部的方位 cv2.imshow("camera", img) if cv2.getWindowProperty('camera', cv2.WND_PROP_VISIBLE) < 1: break # 通过关闭按钮退出程序 cv2.waitKey(1) # if cv2.waitKey(1) & 0xFF == ord("q"): # break # 按下q退出

现在,当我们运行程序后,程序会运行你的计算机默认摄像头,当你露出你的手时,会传出图像圈住你的手部,并且绘制出你的手部主要关节点。

其中,你的手部主要关节点已经标好序号,你的手部分为了21个关节点,指尖分别为4 8 12 16 20

具体关节分为:

手势识别方法

通过前面的讲解,我们完成了手部获取与识别、识别内容的输入,那么我们现在就来开始写我们的手势识别方法。这里,我们用到识别模块里的fingersUp()方法。

找到我们刚刚写的Main.py文件(识别内容输入方法),当我们找到并绘制出我们的手部位置以后,此时的findPosition()方法会得到你的手部具体方位,其中lmList是关节位置方位(type:list),bbox是边框方位(type:dict),当未识别到内容时两者均为空。所以,我们只需要写当数组中存在数据(非空),进行手指判断即可,那么我们可以写成

# -*- coding:utf-8 -*-"""CODE >>> SINCE IN CAIXYPROMISE.MOTTO >>> STRIVE FOR EXCELLENT.CONSTANTLY STRIVING FOR SELF-IMPROVEMENT.@ By: CaixyPromise@ Date: 2021-10-17"""def Gesture_recognition(self): while True: self.detector = HandDetector() frame, img = self.camera.read() img = self.detector.findHands(img) lmList, bbox = self.detector.findPosition(img) if lmList: x1, x2, x3, x4, x5 = self.detector.fingersUp()

上面我们fingersUp()方法谈到,fingersUp()方法会传回从大拇指开始数的长度为5的数组,立起的手指标记为1,放下标记为0。

本次我们的目的时写一个识别我们生活常见的数字手势以及一个赞扬大拇指的手势。结合我们生活,识别你的手势可以写为

# -*- coding:utf-8 -*-"""CODE >>> SINCE IN CAIXYPROMISE.MOTTO >>> STRIVE FOR EXCELLENT.CONSTANTLY STRIVING FOR SELF-IMPROVEMENT.@ By: CaixyPromise@ Date: 2021-10-17"""def Gesture_recognition(self): while True: self.detector = HandDetector() frame, img = self.camera.read() img = self.detector.findHands(img) lmList, bbox = self.detector.findPosition(img) if lmList: x1, x2, x3, x4, x5 = self.detector.fingersUp() if (x2 == 1 and x3 == 1) and (x4 == 0 and x5 == 0 and x1 == 0): # TWO elif (x2 == 1 and x3 == 1 and x4 == 1) and (x1 == 0 and x5 == 0): # THREE elif (x2 == 1 and x3 == 1 and x4 == 1 and x5 == 1) and (x1 == 0): # FOUR elif x1 == 1 and x2 == 1 and x3 == 1 and x4 == 1 and x5 == 1: # FIVE elif x2 == 1 and (x1 == 0, x3 == 0, x4 == 0, x5 == 0): # ONE elif x1 and (x2 == 0, x3 == 0, x4 == 0, x5 == 0): # NICE_GOOD

完成基本的识别以后,我们要把内容表达输出出来。这里我们结合bbox返回的手部方框方位,再使用opencv里的putText方法,实现识别结果的输出。

# -*- coding:utf-8 -*-"""CODE >>> SINCE IN CAIXYPROMISE.MOTTO >>> STRIVE FOR EXCELLENT.CONSTANTLY STRIVING FOR SELF-IMPROVEMENT.@ By: CaixyPromise@ Date: 2021-10-17"""def Gesture_recognition(self): while True: self.detector = HandDetector() frame, img = self.camera.read() img = self.detector.findHands(img) lmList, bbox = self.detector.findPosition(img) if lmList: x_1, y_1 = bbox["bbox"][0], bbox["bbox"][1] x1, x2, x3, x4, x5 = self.detector.fingersUp() if (x2 == 1 and x3 == 1) and (x4 == 0 and x5 == 0 and x1 == 0): cv2.putText(img, "2_TWO", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif (x2 == 1 and x3 == 1 and x4 == 1) and (x1 == 0 and x5 == 0): cv2.putText(img, "3_THREE", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif (x2 == 1 and x3 == 1 and x4 == 1 and x5 == 1) and (x1 == 0): cv2.putText(img, "4_FOUR", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif x1 == 1 and x2 == 1 and x3 == 1 and x4 == 1 and x5 == 1: cv2.putText(img, "5_FIVE", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif x2 == 1 and (x1 == 0, x3 == 0, x4 == 0, x5 == 0): cv2.putText(img, "1_ONE", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif x1 and (x2 == 0, x3 == 0, x4 == 0, x5 == 0): cv2.putText(img, "GOOD!", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) cv2.imshow("camera", img) if cv2.getWindowProperty('camera', cv2.WND_PROP_VISIBLE) < 1: break cv2.waitKey(1)

现在,我们已经完成手势的识别与结果输出,我们把完整代码运行一下,即可验证出我们的代码效果。

完整代码

完整代码如下

# -*- coding:utf-8 -*-"""CODE >>> SINCE IN CAIXYPROMISE.STRIVE FOR EXCELLENT.CONSTANTLY STRIVING FOR SELF-IMPROVEMENT.@ by: caixy@ date: 2021-10-1"""import cv2from HandTrackingModule import HandDetectorclass Main: def __init__(self): self.camera = cv2.VideoCapture(0,cv2.CAP_DSHOW) self.camera.set(3, 1280) self.camera.set(4, 720) def Gesture_recognition(self): while True: self.detector = HandDetector() frame, img = self.camera.read() img = self.detector.findHands(img) lmList, bbox = self.detector.findPosition(img) if lmList: x_1, y_1 = bbox["bbox"][0], bbox["bbox"][1] x1, x2, x3, x4, x5 = self.detector.fingersUp() if (x2 == 1 and x3 == 1) and (x4 == 0 and x5 == 0 and x1 == 0): cv2.putText(img, "2_TWO", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif (x2 == 1 and x3 == 1 and x4 == 1) and (x1 == 0 and x5 == 0): cv2.putText(img, "3_THREE", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif (x2 == 1 and x3 == 1 and x4 == 1 and x5 == 1) and (x1 == 0): cv2.putText(img, "4_FOUR", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif x1 == 1 and x2 == 1 and x3 == 1 and x4 == 1 and x5 == 1: cv2.putText(img, "5_FIVE", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif x2 == 1 and (x1 == 0, x3 == 0, x4 == 0, x5 == 0): cv2.putText(img, "1_ONE", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) elif x1 and (x2 == 0, x3 == 0, x4 == 0, x5 == 0): cv2.putText(img, "GOOD!", (x_1, y_1), cv2.FONT_HERSHEY_PLAIN, 3, (0, 0, 255), 3) cv2.imshow("camera", img) if cv2.getWindowProperty('camera', cv2.WND_PROP_VISIBLE) < 1: break cv2.waitKey(1) # if cv2.waitKey(1) & 0xFF == ord("q"): # breakif __name__ == '__main__': Solution = Main() Solution.Gesture_recognition()

效果一目了然,计算机成功识别了你的手势并把内容输出。快去试试吧!

结语

本篇计算机视觉的手势识别内容写完了,这也是本人第一篇关于人工智能类计算机视觉的推文,后续我们也会持续输出有关于人工智能类的文章,如果本篇写作有纰漏和错误或疑问的地方,还望各位能在评论区指出,让我们一起共同进步一起学习。

创作不易,如果你觉得这篇文章对你有用的话,别忘了点赞在看+关注噢!

下一篇我们将介绍手势识别的进阶——动态手势的识别,我们会将文章第一时间发送在微信公众号:“01编程小屋”当中,别忘了关注我们的公众号以免错过了噢!

本文链接地址:https://www.jiuchutong.com/zhishi/300850.html 转载请保留说明!

上一篇:Python 爬虫案例(python爬虫案例题目)

下一篇:自动驾驶入门必须要学会的ADAS(详解)(自动驾驶科普)

  • 三星手机虚拟内存怎么设置(三星手机虚拟内存4GB好还是8)

    三星手机虚拟内存怎么设置(三星手机虚拟内存4GB好还是8)

  • 华为手机老年模式怎么调(华为智能手机老年模式)

    华为手机老年模式怎么调(华为智能手机老年模式)

  • 能不能通过微信名字搜到人(能不能通过微信支付加对方好友)

    能不能通过微信名字搜到人(能不能通过微信支付加对方好友)

  • 荣耀x10后盖是什么材质(荣耀x10的后盖)

    荣耀x10后盖是什么材质(荣耀x10的后盖)

  • 华为手机怎么关闭手机屏保(华为手机怎么关闭运行的程序)

    华为手机怎么关闭手机屏保(华为手机怎么关闭运行的程序)

  • 苹果没有钱却支付成功(苹果没有钱为什么可以支付)

    苹果没有钱却支付成功(苹果没有钱为什么可以支付)

  • 苹果换过外屏爱思能看出来吗(苹果换过外屏爱思能检查出来吗)

    苹果换过外屏爱思能看出来吗(苹果换过外屏爱思能检查出来吗)

  • 无法访问个人文件夹(无法访问个人文件夹是否自动修复个人文件夹权限)

    无法访问个人文件夹(无法访问个人文件夹是否自动修复个人文件夹权限)

  • k30pro有耳机孔吗(redmik30pro有耳机孔吗)

    k30pro有耳机孔吗(redmik30pro有耳机孔吗)

  • 荣耀30pro有红外功能吗(荣耀30pro有红外线功能吗)

    荣耀30pro有红外功能吗(荣耀30pro有红外线功能吗)

  • 复制图片快捷键ctrl加什么(在电脑上怎么任意截图)

    复制图片快捷键ctrl加什么(在电脑上怎么任意截图)

  • hsyu6e是什么网线(6类网线的标识)

    hsyu6e是什么网线(6类网线的标识)

  • 微信朋友圈背景怎么恢复默认(微信朋友圈背景图可以不让别人看到么)

    微信朋友圈背景怎么恢复默认(微信朋友圈背景图可以不让别人看到么)

  • 支付宝怎么解除拉黑好友(支付宝怎么解除实名认证)

    支付宝怎么解除拉黑好友(支付宝怎么解除实名认证)

  • 抖音号怎么改纯数字(抖音号怎么改纯数字id2022)

    抖音号怎么改纯数字(抖音号怎么改纯数字id2022)

  • 平板电脑上怎么下载钉钉(平板电脑上怎么下载软件)

    平板电脑上怎么下载钉钉(平板电脑上怎么下载软件)

  • 华为小艺可以改名吗(华为小艺可以改语言吗)

    华为小艺可以改名吗(华为小艺可以改语言吗)

  • wps文本转换为数字(wps文本转换为数字怎么做)

    wps文本转换为数字(wps文本转换为数字怎么做)

  • 美团取消的订单怎么恢复(美团取消的订单可以评价吗)

    美团取消的订单怎么恢复(美团取消的订单可以评价吗)

  • word域代码怎么设置页码(word域代码怎么保留两位小数)

    word域代码怎么设置页码(word域代码怎么保留两位小数)

  • qq好友被删了怎么恢复(qq好友被删了怎么辅助验证)

    qq好友被删了怎么恢复(qq好友被删了怎么辅助验证)

  • 面试谈jvm原理(jvm原理面试题)

    面试谈jvm原理(jvm原理面试题)

  • qq离线留言自动回复(qq离线的留言)

    qq离线留言自动回复(qq离线的留言)

  • 虾米音乐如何切歌(虾米音乐怎么用耳机切歌)

    虾米音乐如何切歌(虾米音乐怎么用耳机切歌)

  • 冰岛羊 (© John Porter LRPS/Alamy)

    冰岛羊 (© John Porter LRPS/Alamy)

  • 比赛奖金要交税么
  • 生产成本科目期末余额反映的是
  • 采购砂石料无发票对税务有影响
  • 成品油认证后要开票怎么做
  • 利润表研发费用包括哪些内容
  • 简易征收应纳税额计算
  • 个人开劳务发票几个点
  • 企业一般每个月几号上工伤保险
  • 远程认证是什么意思
  • 增值税发票校验码是什么意思
  • 回迁安置房税收优惠
  • 常用的索赔费用计算方法
  • 进项税额大于销项税额怎么办
  • 怎么解决笔记本电脑卡顿问题
  • 没有发票申报纳税怎么办
  • deepin安装windows字体
  • msoxmled.exe是什么软件
  • 协调费用应该怎么表述才合理
  • 绿萝 用什么土
  • 金融资产终止确认是什么意思
  • php编程技术
  • 年末结转利润分配账户的借方余额表示
  • 进项税额不得抵扣的情况有哪些
  • 业务招待费用列支范围
  • 堪察加半岛上的汉族人
  • 微信小程序自定义tabbar
  • web安全什么意思
  • 设计公司发生的费用
  • 国税代开专票缴纳城建税分录
  • 无偿划转资产涉及产权转移应交税
  • js继承的几种方式 各有什么优缺点
  • 一次性收取一年租金增值税怎么计算
  • 什么样的资产可以执行
  • 缴纳印花税需要带公章吗
  • mysql建表的完整步骤
  • 收到境外服务费会计分录
  • 财务报表的作用包括?
  • MicrosoftSQLserver2014可以卸载吗
  • 开增值税发票规格是否可以不用填?
  • 私人借款条怎么写合法
  • 应收账款项目期末余额怎么计算
  • 进项已抵扣,退货发票怎么处理
  • 用友T3怎么结转成本费用
  • 公司暂估成本分录
  • 鉴证咨询服务费可以抵扣吗
  • 股东的义务是什么意思?
  • 限售股包括哪些
  • sql server分组查询
  • 自动软件脚本
  • windowsserver2008r2standard激活
  • 没有光驱启动
  • 启动mac问号文件夹闪烁怎么办
  • centos7 samba服务器搭建
  • u盘装系统系统资料会被泄露吗
  • win8.1系统安装
  • crossfire.exe是什么
  • WIN7系统如何设置表格默认保存位置
  • win8打不开软件怎么解决
  • xp3软件
  • WIN7系统安装
  • git连接linux服务器
  • win10系统如何清洗打印机喷头
  • win8怎么改文件格式
  • linux查看rpm是否安装
  • linuxdhcp服务器管理与配置
  • perl中\s+
  • cocos2dx游戏有哪些
  • linux执行多个脚本
  • 批处理 ftp
  • opengl shader实例
  • centos6升级到centos8
  • python写汉字代码
  • context和getApplicationContext()介绍
  • python设计gui
  • java script课程
  • 安卓手机管家删除的照片怎么恢复
  • javascript面向对象 第三方类库
  • 广东省国家税务总局稽查局局长
  • 增值税纳税申报操作流程
  • 税务局风险防控形成长远
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设