位置: IT常识 - 正文

2022.07.25 C++下使用opencv部署yolov7模型(五)(c++~怎么用)

编辑:rootadmin
2022.07.25 C++下使用opencv部署yolov7模型(五) 0.写在最前

推荐整理分享2022.07.25 C++下使用opencv部署yolov7模型(五)(c++~怎么用),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:c++ kbhit,c++ ...,c++ ...,c++中使用c,c++%怎么用,c++2019怎么用,c++ ...,c++ ...,内容如对您有帮助,希望把文章链接给更多的朋友!

此篇文字针对yolov7-1.0版本。

最近粗略的看了一遍yolov7的论文,关于yolov7和其他yolo系列的对比,咱就不多说了,大佬们的文章很多很详细。关于opencv部署方面,其实yolov7和yolov5的初期版本(5.0以前的版本)很像,分为三个输出口,yolov5-6.0之后的版本合并了三个输出口变成一个output输出【需要注意的是,虽然yolov可以在export的时候加上--grid参数将detect层加入之后变成和yolov5最新版本的输出一致(可以不用改yolov5代码直接跑yolov7的那种一致,当然,anchors数据还是得改的),但是我试过了,opencv包括onnxruntime推理加grid参数的onnx模型都有问题,暂时我也在探索一种适用于所有yolov7版本的修改方案,但是改了几种都是适用某几个模型,其他模型挂掉的情况】。使用Netron打开两个模型对比下很明显,数据格式也和yolo的一致。所以基本上可以和yolov5的代码通用。只不过具体使用的时候还是有一点区别的。另外,yolov7目前可以直接通过其自身带的export.py导出onnx模型,并不需要像yolov5早期的代码修改。

一.yolov5代码修改适用yolov7

1.归一化框的读取类似yolov5的早期版本

上面说过,yolov7和yolov5的不同,实际上应该是一致的才对(实际上,如果yolov7导出的时候加上--grid参数,结果就和yolov5目前的版本一毛一样,但是加上之后opencv推理onnx的时候会报错,目前yolov7暂时未修复该bug,所以下面的yolov7代码导出的时候不要加--grid参数)。我没仔细debug,所以我们需要根据下面的红色框中的内容对网络的归一化anchors框进行变换变成正常的像素位置。也就是像yolov5之前古老的版本没优化之前一样(这就是我上面说的和yolov5-5.0以前的版本类似的原因)。可以看第三篇的代码中的读取归一化框的方式获取原始图像位置。2021.09.02更新说明 c++下使用opencv部署yolov5模型 (三)_爱晚乏客游的博客-CSDN博客_c++ yolov5

2.anchors数据不同2022.07.25 C++下使用opencv部署yolov7模型(五)(c++~怎么用)

 对比下两者的anchors数据,可以看到两个的anchors不一致了,修改这部分内容即可。

所以综上所诉,对于yolov5-6.0的代码,修改一些地方即可马上应用到yolov7上面,可以说很方便了。

具体修改有两处,一处是anchors,另外一处是推理程序,修改之后的链接我放最下面了,其实就是在第四篇的基础上面修改下:GitHub - UNeedCryDear/yolov5-opencv-dnn-cpp: 使用opencv模块部署yolov5-6.0版本 

//yolo.h中改下anchorsconst float netAnchors[3][6] = { {12, 16, 19, 36, 40, 28},{36, 75, 76, 55, 72, 146},{142, 110, 192, 243, 459, 401} }; //yolov7-P5 anchors//yolo.cpp中推理代码修改bool Yolo::Detect(Mat& SrcImg, Net& net, vector<Output>& output) {Mat blob;int col = SrcImg.cols;int row = SrcImg.rows;int maxLen = MAX(col, row);Mat netInputImg = SrcImg.clone();if (maxLen > 1.2 * col || maxLen > 1.2 * row) {Mat resizeImg = Mat::zeros(maxLen, maxLen, CV_8UC3);SrcImg.copyTo(resizeImg(Rect(0, 0, col, row)));netInputImg = resizeImg;}vector<Ptr<Layer> > layer;vector<string> layer_names;layer_names= net.getLayerNames();blobFromImage(netInputImg, blob, 1 / 255.0, cv::Size(netWidth, netHeight), cv::Scalar(0, 0, 0), true, false);//如果在其他设置没有问题的情况下但是结果偏差很大,可以尝试下用下面两句语句//blobFromImage(netInputImg, blob, 1 / 255.0, cv::Size(netWidth, netHeight), cv::Scalar(104, 117, 123), true, false);//blobFromImage(netInputImg, blob, 1 / 255.0, cv::Size(netWidth, netHeight), cv::Scalar(114, 114,114), true, false);net.setInput(blob);std::vector<cv::Mat> netOutputImg;net.forward(netOutputImg, net.getUnconnectedOutLayersNames());std::vector<int> classIds;//结果id数组std::vector<float> confidences;//结果每个id对应置信度数组std::vector<cv::Rect> boxes;//每个id矩形框float ratio_h = (float)netInputImg.rows / netHeight;float ratio_w = (float)netInputImg.cols / netWidth;int net_width = className.size() + 5; //输出的网络宽度是类别数+5for (int stride = 0; stride < strideSize; stride++) { //stridefloat* pdata = (float*)netOutputImg[stride].data;int grid_x = (int)(netWidth / netStride[stride]);int grid_y = (int)(netHeight / netStride[stride]);for (int anchor = 0; anchor < 3; anchor++) {//anchorsconst float anchor_w = netAnchors[stride][anchor * 2];const float anchor_h = netAnchors[stride][anchor * 2 + 1];for (int i = 0; i < grid_y; i++) {for (int j = 0; j < grid_x; j++) {float box_score = sigmoid_x(pdata[4]); ;//获取每一行的box框中含有某个物体的概率if (box_score >= boxThreshold) {cv::Mat scores(1, className.size(), CV_32FC1, pdata + 5);Point classIdPoint;double max_class_socre;minMaxLoc(scores, 0, &max_class_socre, 0, &classIdPoint);max_class_socre = sigmoid_x(max_class_socre);if (max_class_socre >= classThreshold) {float x = (sigmoid_x(pdata[0]) * 2.f - 0.5f + j) * netStride[stride]; //xfloat y = (sigmoid_x(pdata[1]) * 2.f - 0.5f + i) * netStride[stride]; //yfloat w = powf(sigmoid_x(pdata[2]) * 2.f, 2.f) * anchor_w; //wfloat h = powf(sigmoid_x(pdata[3]) * 2.f, 2.f) * anchor_h; //hint left = (int)(x - 0.5 * w) * ratio_w + 0.5;int top = (int)(y - 0.5 * h) * ratio_h + 0.5;classIds.push_back(classIdPoint.x);confidences.push_back(max_class_socre * box_score);boxes.push_back(Rect(left, top, int(w * ratio_w), int(h * ratio_h)));}}pdata += net_width;//下一行}}}}//执行非最大抑制以消除具有较低置信度的冗余重叠框(NMS)vector<int> nms_result;NMSBoxes(boxes, confidences, nmsScoreThreshold, nmsThreshold, nms_result);for (int i = 0; i < nms_result.size(); i++) {int idx = nms_result[i];Output result;result.id = classIds[idx];result.confidence = confidences[idx];result.box = boxes[idx];output.push_back(result);}if (output.size())return true;elsereturn false;}二、yolov7一些模型转换的问题

评论区有些小伙伴反馈yolov7-d6模型在opencv4.5.1下面会报错,报错信息类似之前读取早期的yolov5的报错一致。

OpenCV(4.5.0) Error: Unspecified error (> Node [Slice]:(341) parse error: OpenCV(4.5.0) D:\opencv\ocv4.5.0\sources\modules\dnn\src\onnx\onnx_importer.cpp:697: error: (-2:Unspecified error) in function 'void __cdecl cv::dnn::dnn4_v20200908::ONNXImporter::handleNode(const class opencv_onnx::NodeProto &)' Slice layer only supports steps = 1 (expected: 'countNonZero(step_blob != 1) == 0'), where 'countNonZero(step_blob != 1)' is 1 must be equal to '0' is 0 in cv::dnn::dnn4_v20200908::ONNXImporter::handleNode, file D:\opencv\ocv4.5.0\sources\modules\dnn\src\onnx\onnx_importer.cpp, line 1797

经过对比yolov7和yolov7-d6的yaml文件,发现是由于yolov7-d6中使用了ReOrg模块引起的报错,也就是步长为2的切片,像我这个系列中第一篇的问题一样。

这个模块有点类似早期的yolov5的Facos模块,需要将ReOrg模块修改成下面的代码。 在models/common.py里面搜索下ReOrg,改成一下代码之后重新导出onnx模型即可正确读取。

class ReOrg(nn.Module): def __init__(self): super(ReOrg, self).__init__() def forward(self, x): # x(b,c,w,h) -> y(b,4c,w/2,h/2) #origin code # return torch.cat([x[..., ::2, ::2], x[..., 1::2, ::2], x[..., ::2, 1::2], x[..., 1::2, 1::2]], 1) self.concat=Contract(gain=2) return self.concat(x)

最后贴个yolov7和yolov5的对比图,可以看到yolov7提升还是蛮明显的,结果的置信度高了一些,后面的自行车也检测出来了,就是领带这里误检了。 

 

贴合github链接,里面包括了yolov7和yolov5,通过宏定义来控制:

GitHub - UNeedCryDear/yolov7-opencv-dnn-cpp

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

上一篇:一文详解Yolov5——基于Yolov5的火灾检测系统(yolov5 入门)

下一篇:Web 攻防之业务安全:接口未授权访问/调用测试(敏感信息泄露)(web攻防之业务安全实战指南)

  • 加拿大失业率微降 核心年龄段女性就业率创新高

    加拿大失业率微降 核心年龄段女性就业率创新高

  • 怎样删除ie缓存(怎样删除ie)(如何彻底删除ie缓存)

    怎样删除ie缓存(怎样删除ie)(如何彻底删除ie缓存)

  • 微信号申请的步骤是怎样的(微信号申请的步数怎么看)

    微信号申请的步骤是怎样的(微信号申请的步数怎么看)

  • 微信置顶文字可以怎么设置(微信置顶文字可以加大字体吗)

    微信置顶文字可以怎么设置(微信置顶文字可以加大字体吗)

  • 华为p30摄像头有几个(华为p30摄像头有灰尘)

    华为p30摄像头有几个(华为p30摄像头有灰尘)

  • 天猫漏发货不承认怎么办(天猫店铺漏发货怎么办)

    天猫漏发货不承认怎么办(天猫店铺漏发货怎么办)

  • 荣耀v30预计上市时间(荣耀v30上市具体时间)

    荣耀v30预计上市时间(荣耀v30上市具体时间)

  • 电话卡可以换身份证吗(电话卡可以换身份证实名吗)

    电话卡可以换身份证吗(电话卡可以换身份证实名吗)

  • 闲鱼卖家不发货钱会退回来吗(闲鱼卖家不发货怎么办)

    闲鱼卖家不发货钱会退回来吗(闲鱼卖家不发货怎么办)

  • 智联招聘账号存在异常(智联招聘账号找不到了怎么办)

    智联招聘账号存在异常(智联招聘账号找不到了怎么办)

  • 苹果x漏液只能换屏幕吗(iphone x漏液还能用多久)

    苹果x漏液只能换屏幕吗(iphone x漏液还能用多久)

  • 企业微信可以看到别人朋友圈吗(企业微信可以看到已读吗)

    企业微信可以看到别人朋友圈吗(企业微信可以看到已读吗)

  • 路由器接口不够用怎么办(路由器接口不够用,怎么添加交换机)

    路由器接口不够用怎么办(路由器接口不够用,怎么添加交换机)

  • 系统软件分为哪四类(系统软件分为哪两种)

    系统软件分为哪四类(系统软件分为哪两种)

  • iphone11左边扩音器没声音(iphone11左边音响)

    iphone11左边扩音器没声音(iphone11左边音响)

  • 苹果11原装充电器多少瓦(苹果11原装充电器多少钱)

    苹果11原装充电器多少瓦(苹果11原装充电器多少钱)

  • 苹果x短信怎么打特效(苹果x短信怎么显示已读)

    苹果x短信怎么打特效(苹果x短信怎么显示已读)

  • 手机摔了一下黑屏了怎么办(手机摔了一下黑屏了但是能震动)

    手机摔了一下黑屏了怎么办(手机摔了一下黑屏了但是能震动)

  • oppo手机自动开机关机(oppo手机自动开启了飞行模式怎么办)

    oppo手机自动开机关机(oppo手机自动开启了飞行模式怎么办)

  • 手机wps绘制表格(手机wps绘制表格怎么操作)

    手机wps绘制表格(手机wps绘制表格怎么操作)

  • 拼多多怎么复制口令到微信(拼多多怎么复制口令)

    拼多多怎么复制口令到微信(拼多多怎么复制口令)

  • 手机号码怎么显示公司名称(手机号码怎么显示空号)

    手机号码怎么显示公司名称(手机号码怎么显示空号)

  • 能量芯片的作用(能量芯片的作用有哪些)

    能量芯片的作用(能量芯片的作用有哪些)

  • 电脑怎么开启vt功能(联想电脑怎么开启vt)

    电脑怎么开启vt功能(联想电脑怎么开启vt)

  • QQ怎么开启聊天记录漫游(qq怎么开启聊天标识)

    QQ怎么开启聊天记录漫游(qq怎么开启聊天标识)

  • beatsx怎么放到收纳盒

    beatsx怎么放到收纳盒

  • 拼多多在哪查订单编号(拼多多哪里查看订单)

    拼多多在哪查订单编号(拼多多哪里查看订单)

  • 怎么防止蹭网呢?(怎么样防止蹭wi-fi)

    怎么防止蹭网呢?(怎么样防止蹭wi-fi)

  • 关于uni-app入门看完这篇就够了(uni-app实例教程)

    关于uni-app入门看完这篇就够了(uni-app实例教程)

  • Vue Admin Template关闭eslint校验,lintOnSave:false设置无效解决办法

    Vue Admin Template关闭eslint校验,lintOnSave:false设置无效解决办法

  • 〖大前端 - 基础入门三大核心之CSS篇㉑〗- 3D变形 与空间移动(前端 大前端)

    〖大前端 - 基础入门三大核心之CSS篇㉑〗- 3D变形 与空间移动(前端 大前端)

  • 分公司需要独立法人吗
  • 其他权益工具影响哪些报表
  • 报销金额大于发票金额几毛钱
  • 减税后含税单价计算公式
  • 公司产品的打样费入什么会计科目
  • 普票销项负数发票
  • 以销定产如何核算成本
  • 怎么识别发票是否含税
  • 一般纳税人利润100万要交多少税
  • 开具银行资信证明
  • 应付账款和长期负债
  • 房屋出租需要缴纳个人所得税吗
  • 个人借款利息是多少才合法
  • 社保证明缴费证明查询在哪查
  • 小规模怎样计算进项税额
  • 地税印花税怎么算
  • 停车费属于不动产租赁服务税率
  • 待摊费用和长期待摊费用的区别
  • 银行支票怎么用
  • 不可抗力后果承担
  • 企业所得税投资收益纳税调整
  • 苹果电脑初始化设置
  • windows11我的电脑怎么放到桌面
  • 进项税大于销项税怎么结转
  • 公司备用金申请单
  • php symlink
  • 烟草企业亏损
  • Realsense D455/435内参标定以及手眼标定
  • 踩雷日记:Pytorch mmcv-full简易安装
  • 坏账准备属于哪一类账户
  • php开源微商城
  • vue叠化在哪里
  • vue动态组件component原理
  • 城市维护建设税减免税优惠政策
  • 孪生神经网络 计算相似度
  • 为什么我的命令提示符里显示user
  • AIGC之GPT-4:GPT-4的简介(核心原理/意义/亮点/技术点/缺点/使用建议)、使用方法、案例应用(计算能力/代码能力/看图能力等)之详细攻略
  • 现代服务增值税纳税义务发生时间
  • 业务预算包括直销费用吗
  • mysql查询性能优化
  • mongodb添加环境变量
  • 预交的企业所得税需要计提吗?
  • 租金收入如何确认收入
  • 固定资产账务处理候文江视频
  • 金税四期的主要内容
  • 差旅费报销原因
  • 为什么合理损耗不计入成本
  • 计提折旧会计分录怎么做
  • 个体工商户要进行汇算清缴吗
  • 不含税转换
  • 打车费会计分录怎么写
  • 劳务报酬如何记账
  • 成本会计难做吗没做过
  • 借款费用应如何入账
  • mysql查询倒数第二个字母为a
  • 选择mysql数据库为当前数据库
  • git checkout撤销
  • porteus中文版下载
  • xp系统能用谷歌吗
  • 命令提示符操作方法
  • a4腰多大
  • nkvmon.exe - nkvmon是什么进程 有什么用
  • 电脑安装win8系统
  • Win10 Build 14316启用Linux Bash 环境图文教程
  • d命令怎么用
  • 安卓listview的用法
  • exact form
  • 被人遗忘的明星
  • cocos2dx4.0入门
  • jquery validator
  • ie6怎么设置兼容性
  • 手机游戏服务器无响应是怎么回事
  • Node.js node-schedule定时任务隔多少分钟执行一次的方法
  • dos基本命令大全关机
  • js函数内置函数
  • powercli命令
  • jQuery使用animate实现ul列表项相互飘动效果示例
  • node运行js文件
  • javascript 触发事件列表 比较不错
  • 股权转让申报期限
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

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

    友情链接: 武汉网站建设