基于yolov5和openpose人体骨骼关键点实现的摔倒姿态识别检测系统实现

【参考源码】

https://github.com/HRonaldo/Openpose_YOLO

本项目参考上面框架进行全面改进,改进如下:

(1)封装YOLOv5检测类,这样可以加强阅读便利性,模块设计方便嵌入其他框架,后面我会换成yolov8模型进行该项目移植,yolov8移植项目请访问:https://blog.csdn.net/FL1623863129/article/details/142413295

(2)调整部分文件夹结构,比如模型归为一类文件夹,这样方便查看模型

(3)检测流程简化,使得人体骨骼关键点实现的摔倒姿态识别代码十分简单,实际只要40多行代码就实现整个检测流程

【效果展示】

【特别注意】

本项目由于使用了3个模型,所以实时性能较差,建议采用GPU推理加快推理速度,同时尽量不要使用cv2.imshow去实时显示窗口,而是采用写出视频结果方式

【测试环境】

代码提供有requirments.txt,但是环境其实比较宽松,比如我的环境是

torch==1.9.0

opencv-python==4.8.0.76

【实现代码】

【原来代码】

detect.py

import argparse
import time
from pathlib import Path
from torch import from_numpy, jitimport cv2
import torch
import torch.backends.cudnn as cudnn
from numpy import random
import os
import datetime
from models.experimental import attempt_load
from utils.datasets import LoadStreams, LoadImages
from utils.general import check_img_size, non_max_suppression, apply_classifier, scale_coords, xyxy2xywh, \strip_optimizer, set_logging, increment_path
from utils.plots import plot_one_box
from utils.torch_utils import select_device, load_classifier, time_synchronizedimport runOpenposeimport YOLOdef detect(save_img=False):global ipsource, weights, view_img, save_txt, imgsz = opt.source, opt.weights, opt.view_img, opt.save_txt, opt.img_sizewebcam = source.isnumeric() or source.endswith('.txt') or source.lower().startswith(('rtsp://', 'rtmp://', 'http://'))# Directoriessave_dir = Path(increment_path(Path(opt.project) / opt.name, exist_ok=opt.exist_ok))  # increment run(save_dir / 'labels' if save_txt else save_dir).mkdir(parents=True, exist_ok=True)  # make dir# 加载摔倒检测的模型print("加载摔倒检测的模型开始")net = jit.load(r'action_detect/checkPoint/openpose.jit')action_net = jit.load(r'action_detect/checkPoint/action.jit')print("加载摔倒检测的模型结束")# Initializeset_logging()# 获取设备device = select_device(opt.device)# 如果设备为gpu,使用Float16half = device.type != 'cpu'  # half precision only supported on CUDA# Load model# 加载Float32模型,确保用户设定的输入图片分辨率能整除32(如果不能则调整为能整除并删除)#model = attempt_load(weights, map_location=device)  # load FP32 modelmodel=YOLO.Predictor(weights,device,opt.classes,opt.conf_thres)imgsz = check_img_size(imgsz, s=32)  # !!!check img_size# 设置Float16if half:model.model.half()  # to FP16# Set Dataloader# 通过不同的输入源来设置不同的数据加载方式vid_path, vid_writer = None, Noneif webcam:view_img = Truecudnn.benchmark = True  # set True to speed up constant image size inferencedataset = LoadStreams(source, img_size=imgsz)else:save_img = True# 如果检测视频的时候想显示出来,可以再这里加一行 view_img = Trueview_img = Truedataset = LoadImages(source, img_size=imgsz)# Get names and colors# 获取类别的名字names = model.module.names if hasattr(model, 'module') else model.model.names# 设置画框的颜色colors = [[random.randint(0, 255) for _ in range(3)] for _ in names]# Run inferencet0 = time.time()# 进行一次向前推理,测试程序是否正常img = torch.zeros((1, 3, imgsz, imgsz), device=device)  # init img_ = model.model(img.half() if half else img) if device.type != 'cpu' else None  # run once"""path 图片/视频路径img 进行resize+pad之后的图片img0 原size图片cap 当读取图片时为None,读取视频时为视频源"""for path, img, im0s, vid_cap in dataset:img = torch.from_numpy(img).to(device)# 图片也设置为Float16img = img.half() if half else img.float()  # uint8 to fp16/32img /= 255.0  # 0 - 255 to 0.0 - 1.0# 没有batch_size 的话则在最前面增加一个轴if img.ndimension() == 3:img = img.unsqueeze(0)# Inferencet1 = time_synchronized()pred=model.predict(source=img)t2 = time_synchronized()# Process detections# 对每一张图片作处理for i, det in enumerate(pred):  # detections per image# 如果输入源是webcam 则batch_size 不为1,取出dataset中的一张图片if webcam:  # batch_size >= 1p, s, im0, frame = path[i], '%g: ' % i, im0s[i].copy(), dataset.countelse:p, s, im0, frame = path, '', im0s, getattr(dataset, 'frame', 0)# print(det)boxList = []  # 框的一个list 给openpose 使用p = Path(p)  # to Path# 设置保存图片/视频的路径save_path = str(save_dir / p.name)  # img.jpg# 设置保存框最薄txt文件的路径txt_path = str(save_dir / 'labels' / p.stem) + ('' if dataset.mode == 'image' else f'_{frame}')  # img.txt# 设置打印信息(图片长宽)s += '%gx%g ' % img.shape[2:]  # print stringgn = torch.tensor(im0.shape)[[1, 0, 1, 0]]  # normalization gain whwhif len(det):# Rescale boxes from img_size to im0 size# 调整预测框的坐标:基于resize+pad的图片的坐标 -->基于原size图片的坐标# 此时坐标格式为xyxydet[:, :4] = scale_coords(img.shape[2:], det[:, :4], im0.shape).round()# Print results# 打印检测到的类别数量for c in det[:, -1].unique():n = (det[:, -1] == c).sum()  # detections per classs += f'{n} {names[int(c)]}s, '  # add to string# Write results# 保存预测结果for *xyxy, conf, cls in reversed(det):if save_txt:  # Write to file# 将xyxy(左上角+右下角)格式转为xywh(中心点+宽长)格式,并除上w,h做归一化,转化为列表再保存xywh = (xyxy2xywh(torch.tensor(xyxy).view(1, 4)) / gn).view(-1).tolist()  # normalized xywhline = (cls, *xywh, conf) if opt.save_conf else (cls, *xywh)  # label formatwith open(txt_path + '.txt', 'a') as f:f.write(('%g ' * len(line)).rstrip() % line + '\n')# 在原图上画框if save_img or view_img:  # Add bbox to imagelabel = f'{names[int(cls)]} {conf:.2f}'plot_one_box(xyxy, im0, label=label, color=colors[int(cls)], line_thickness=3)boxList.append([int(xyxy[0]), int(xyxy[1]), int(xyxy[2]), int(xyxy[3])])runOpenpose.run_demo(net, action_net, [im0], 256, False, boxList)  # 人体姿态检测 将图片和yolov5检测人体的框也传给openpose# Print time (inference + NMS)# 打印向前传播+nms时间print(f'{s}Done. ({t2 - t1:.3f}s)')if save_img:if dataset.mode == 'image':imageName = str(time.strftime('%Y%m%d%H%M%S', time.localtime(time.time()))) + ".jpg"cv2.imwrite(save_path+imageName, im0)else:  # 'video'if vid_path != save_path:  # new videovid_path = save_pathif isinstance(vid_writer, cv2.VideoWriter):vid_writer.release()  # release previous video writerfourcc = 'mp4v'  # output video codecfps = vid_cap.get(cv2.CAP_PROP_FPS)w = int(vid_cap.get(cv2.CAP_PROP_FRAME_WIDTH))h = int(vid_cap.get(cv2.CAP_PROP_FRAME_HEIGHT))vid_writer = cv2.VideoWriter(save_path, cv2.VideoWriter_fourcc(*fourcc), fps, (w, h))vid_writer.write(im0)print('保存图片')# 打印总时间print(f'Done. ({time.time() - t0:.3f}s)')cv2.destroyAllWindows()if __name__ == '__main__':parser = argparse.ArgumentParser()# 选用训练的权重,可用根目录下的yolov5s.pt,也可用runs/train/exp/weights/best.ptparser.add_argument('--weights', type=str, default='models/hub/yolov8s.pt', help='model.pt path(s)')# 检测数据,可以是图片/视频路径,也可以是'0'(电脑自带摄像头),也可以是rtsp等视频流parser.add_argument('--source', type=str, default='0',help='source')  # file/folder, 0 for webcam# 网络输入图片大小parser.add_argument('--img-size', type=int, default=640, help='inference size (pixels)')# 置信度阈值,检测到的对象属于特定类(狗,猫,香蕉,汽车等)的概率parser.add_argument('--conf-thres', type=float, default=0.5, help='object confidence threshold')# 做nms的iou阈值parser.add_argument('--iou-thres', type=float, default=0.45, help='IOU threshold for NMS')# 检测的设备,cpu;0(表示一个gpu设备cuda:0);0,1,2,3(多个gpu设备)。值为空时,训练时默认使用计算机自带的显卡或CPUparser.add_argument('--device', default='gpu', help='cuda device, i.e. 0 or 0,1,2,3 or cpu')# 是否展示检测之后的图片/视频,默认Falseparser.add_argument('--view-img', action='store_true', help='display results')# 是否将检测的框坐标以txt文件形式保存,默认Falseparser.add_argument('--save-txt', action='store_true', help='save results to *.txt')# 是否将检测的labels以txt文件形式保存,默认Falseparser.add_argument('--save-conf', action='store_true', help='save confidences in --save-txt labels')# 设置只保留某一部分类别,如0或者0 2 3parser.add_argument('--classes', nargs='+', default=[0,67], type=int,help='filter by class: --class 0, or --class 0 2 3')# 进行nms是否也去除不同类别之间的框,默认Falseparser.add_argument('--agnostic-nms', action='store_true', help='class-agnostic NMS')# 推理的时候进行多尺度,翻转等操作(TTA)推理parser.add_argument('--augment', action='store_true', help='augmented inference')# 如果为True,则对所有模型进行strip_optimizer操作,去除pt文件中的优化器等信息,默认为Falseparser.add_argument('--update', action='store_true', help='update all models')# 检测结果所存放的路径,默认为runs/detectparser.add_argument('--project', default='runs/detect', help='save results to project/name')# 检测结果所在文件夹的名称,默认为expparser.add_argument('--name', default='exp', help='save results to project/name')# 若现有的project/name存在,则不进行递增parser.add_argument('--exist-ok', action='store_true', help='existing project/name ok, do not increment')opt = parser.parse_args()print(opt)with torch.no_grad():detect()

我的代码

main_video.py

from torch import from_numpy, jit
from Yolov5Manager import *
import runOpenpose
import cv2device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
# 加载摔倒检测的模型
print("加载摔倒检测的模型开始")
net = jit.load(r'weights/openpose.jit', map_location=device)
action_net = jit.load(r'weights/action.jit', map_location=device)
print("加载yolov5模型")
detector = Yolov5Manager()
print('所有模型加载完成!')
video_file = "video/tt.mp4"
save_file = 'result.mp4'
cap = cv2.VideoCapture(video_file)# 获取视频帧速率 FPS
frame_fps = int(cap.get(cv2.CAP_PROP_FPS))
# 获取视频帧宽度和高度
frame_width = int(cap.get(cv2.CAP_PROP_FRAME_WIDTH))
frame_height = int(cap.get(cv2.CAP_PROP_FRAME_HEIGHT))
print("video fps={},width={},height={}".format(frame_fps, frame_width, frame_height))
fourcc = cv2.VideoWriter_fourcc(*'mp4v')
out = cv2.VideoWriter(save_file, fourcc, frame_fps, (frame_width, frame_height))
count = 0while True:ret, frame = cap.read()if not ret:print('read over!')breakcount += 1result_lists = detector.inference_image(frame)for box in result_lists:x = box[4] - box[2]y = box[3] - box[1]if x / y >= 0.8:  # 比例>0.8 可能会是摔倒print('进行人体姿态检测')box_list = [res[2:6] for res in result_lists]runOpenpose.run_demo(net, action_net, [frame], 256, False, box_list)  # 人体姿态检测 将图片和yolov5检测人体的框也传给openposebreak#frame = detector.draw_image(frame,result_list=result_lists)#out.write(frame)print('detected frame {}'.format(count))cv2.imshow('frame',frame)if cv2.waitKey(1) & 0xFF == ord('q'):break
out.release()
cap.release()
cv2.destroyAllWindows()

如果您想把人画上去,请打开注释   

第52行 #frame = detector.draw_image(frame,result_list=result_lists)

如果您保存视频,请打开注释

第53行    #out.write(frame)

【完整源码下载地址】

https://download.csdn.net/download/FL1623863129/89777684

【延伸扩展】

运行runOpenpose.py
只跑了open pose 可以获得人体的关键点图,用于后续的.jit模型训练 人体的关键点图会保存在data/test中 pose.py中draw方法的最下面可以控制保存关键点图的位置

如果想要检测其他姿势: 1.收集图片,跑runOpenpose.py 文件获得人体的关键点图 2.对人体的关键点图根据自己想要的进行分类放在data/train 和 data/test 3.跑 action_detect/train.py

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1541338.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

2. qgis c++ api 整体框架详解

转载 2. qgis c api 整体框架详解-CSDN博客 整体架构 QGis库官方文档 下表是官方文档中的模块说明: 其中3D,QgsQuick和server库需要在编译之前进行配置,配置项分别为WITH_3D WITH_QUICK和WITH_SERVER,具体编译配置方法见开发环…

祝天下老师教师节快乐-HTML+java script

效果图 &#x1f680;HTML结构揭秘&#x1f680;: &#x1f4dd;<head>中包含了元信息和样式表。 &#x1f5bc;️<body>拥有一个<canvas>元素与一个.text类的<div>&#xff0c;为星空与教师节信息搭建舞台。 &#x1f3a8;CSS魔法调色盘&#x1f3a…

计算机网络:物理层 --- 基本概念、编码与调制

目录 一. 物理层的基本概念 二. 数据通信系统的模型 三. 编码 3.1 基本概念 3.2 不归零制编码 3.3 归零制编码 3.4 曼切斯特编码 3.5 差分曼切斯特编码 ​编辑 四. 调制 4.1 调幅 4.2 调频 4.3 调相 4.4 混合调制 今天我们讲的是物理…

利士策分享,中日核污染水排海问题共识背后的深思

利士策分享&#xff0c;中日核污染水排海问题共识背后的深思 近日&#xff0c;中日两国在应对福岛第一核电站核污染水排放问题上达成了重要共识&#xff0c;这一进展无疑值得肯定。 然而&#xff0c;这背后所引发的关于金钱、责任与利益的讨论&#xff0c;却值得我们深入探究。…

【解决】chrome 谷歌浏览器,鼠标点击任何区域都是 Input 输入框的状态,能看到输入的光标

chrome 谷歌浏览器&#xff0c;鼠标点击任何区域都是 Input 输入框的状态&#xff0c;能看到输入的光标 今天打开电脑的时候&#xff0c;网页中任何文本的地方&#xff0c;只要鼠标点击&#xff0c;就会出现一个输入的光标&#xff0c;无论在哪个站点哪个页面都是如此。 我知道…

【练习15】拼三角

链接&#xff1a;A-拼三角_牛客小白月赛32 (nowcoder.com) 分析&#xff1a; 三角形的三个边a、b、c必须满足条件&#xff1a;ab>c,ac>b,bc>a 可以先将输入的六个数按从小到大排序&#xff0c;然后再枚举。 因为已经排好序了&#xff0c;如果第一行ab<c&#xff0c…

YOLOv8改进,YOLOv8替换主干网络为VanillaNet( CVPR 2023 华为提出的全新轻量化架构),大幅度涨点

改进前训练结果: 改进后训练结果: 摘要 基础模型的核心理念是“更多即不同”,这一理念在计算机视觉和自然语言处理领域取得了惊人的成功。然而,变压器模型的优化挑战和固有复杂性呼唤一种向简化转变的范式。在本研究中,引入了VanillaNet,一种拥抱设计优雅的神经网络架构…

2024年中国研究生数学建模竞赛A题“风电场有功功率优化分配”全析全解

问题一&#xff1a; 针对问题一&#xff0c;可以采用以下低复杂度模型&#xff0c;来计算风机主轴及塔架的疲劳损伤累积程度。 建模思路&#xff1a; 累积疲劳损伤计算&#xff1a; 根据Palmgren-Miner线性累积损伤理论&#xff0c;元件的疲劳损伤可以累积。因此&#xff0c;…

基于SpringBoot+Vue的商城积分系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 精品专栏&#xff1a;Java精选实战项目源码、Python精…

回归预测 | Matlab实现INFO-HKELM向量加权算法优化混合核极限学习机多变量回归预测

回归预测 | Matlab实现INFO-HKELM向量加权算法优化混合核极限学习机多变量回归预测 目录 回归预测 | Matlab实现INFO-HKELM向量加权算法优化混合核极限学习机多变量回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现INFO-HKELM向量加权算法优化混合核极…

JAVA连接世界驾驭未来国际版二手车交易市场系统小程序源码

&#x1f697;【连接世界&#xff0c;驾驭未来 —— 探索国际版二手车交易市场系统】&#x1f697; &#x1f30d; 开篇&#xff1a;梦想无界&#xff0c;车行天下 &#x1f30d; 在这个日新月异的时代&#xff0c;我们对未来的憧憬不再局限于脚下的土地。你是否曾梦想过&…

【d46】【Java】【力扣】876.链表的中间结点

思路 先获得总体长度&#xff0c; 再得到中间节点 的索引&#xff0c;&#xff0c;这里的索引是从1开始的索引&#xff0c;而不是从0开始的索引(这种理解方式更简单) 排错&#xff1a;另一个思路&#xff1a;将链表都放进list&#xff0c;获得中间的数字&#xff0c;然后遍历…

【滑动窗口】算法总结

文章目录 滑动窗口算法总结1.暴力求解vs滑动窗口2.需要注意的细节问题 2.滑动窗口的基本模板1.非固定窗口大小的滑动窗口2.固定窗口大小的滑动窗口细节 滑动窗口算法总结 1.暴力求解vs滑动窗口 遇到那些可以转化成一个子数组的长度的问题时&#xff0c;往往需要用到双指针。 …

(undone) 声音信号处理基础知识(2)

来源&#xff1a;https://www.youtube.com/watch?vbnHHVo3j124 复习物理知识&#xff1a; 声音由物体的振动产生 物体振动会导致空气分支振荡 某一处的空气气压变化会创造一个波 声音是机械波 空气的振荡在空间中传递 能量从空间中的一个点到另一个点 机械波需要媒介&#x…

中国电子学会202406青少年软件编程(Python)等级考试试卷(四级)真题

青少年软件编程(Python)等级考试试卷(四级)2024-6 一、单选题(共25题, 共50分) 1.执行以下程序后所输出的结果是?( ) A 20 B 41 C 21 D 91 2.以下说法错误的是?( ) A python中可以在不同的自定义函数中声明相同名字的变量,使用时不会造成数据混…

前端框架Vue、React、Angular、Svelte对比

在对比 React、Vue.js、Angular 和 Svelte 时&#xff0c;除了在高层次的特性上有显著差异&#xff0c;它们在核心设计理念和底层实现机制上也有明显的不同。为了清晰地理解这些框架&#xff0c;我们可以从以下几个方面来分析它们的核心不同点和底层不同点。 1. 框架类型和设计…

【学习笔记】STM32F407探索者HAL库开发(四)F103时钟系统配置

【学习笔记】STM32F407探索者HAL库开发&#xff08;四&#xff09;F103时钟系统配置 1 STM32F1时钟树1.1 STM32F103时钟系统图1.2 STM32F103时钟树简图1.2.1 高速部分1.2.2 低速部分 1.3 函数配置1.4 时钟输出1.5 STM32CubeMX时钟树配置F11.6 时钟系统对与嵌入式开发的重要性 1…

Qt-QLabel 添加图片并设置 GIF 图动态效果

Qt-QLabel 添加图片并设置 GIF 图动态效果 一、添加图片资源并设置图片 选择标签&#xff0c;拖拉到界面上&#xff0c;然后选择器属性 picmap   选择设置&#xff0c;在这里添加图片资源   点击左边的加号符号按钮添加前缀&#xff0c;并设置前缀名&#xff0c;如果已经…

Nginx静态资源优化、压缩、缓存处理

一、静态资源优化配置语法 Nginx对静态资源如何进行优化配置。这里从三个属性配置进行优化&#xff1a; sendfile on; tcp_nopush on; tcp_nodeplay on; &#xff08;1&#xff09;sendfile&#xff0c;用来开启高效的文件传输模式。 语法sendfile on |off;默认值sendfile …

奥比中光深度相机相关使用内容

奥比中光深度相机相关使用内容 Windows平台测试官方软件关于python环境的配置1、安装CMake2、安装Visual Studio3、项目地址下载4、配置Visual Studio5、完成基于Python的SDK配置官网教学视频地址 3D视觉开发者社区 官方文档地址 效果: Windows平台测试官方软件 Window…