yolov5-6.0 tensorrt推理

CMakeLists.txt

cmake_minimum_required(VERSION 3.0)		# CMake最低版本要求,低于2.6的构建过程会被终止set(CMAKE_CXX_STANDARD 14)		#opencv4以上加此句
set(CMAKE_CXX_STANDARD  14)
project(cmake_test)				# 定义工程名称
find_package(OpenCV     REQUIRED)
# find_package(PCL 	    REQUIRED)
find_package(CUDA       REQUIRED)
find_package(TensorRT 	REQUIRED)
include_directories(${PROJECT_SOURCE_DIR}/include# ${PCL_INCLUDE_DIRS} ${CUDA_INCLUDE_DIRS} ${TensorRT_INCLUDE_DIRS}${TensorRT_INCLUDE_DIRS}/../samples/common)
add_executable(infer src/main.cpp)target_link_libraries( infer ${OpenCV_LIBS} # ${PCL_LIBRARIES} ${CUDA_LIBRARIES} ${TensorRT_LIBS}) # 链接库

infer.h

#include <iostream>
#include <vector>
#include <list>
using namespace std;string names[] = {"xxxxx"};     // 修改自己的标签struct BOX
{float x;float y;float width;float height;
};struct Object
{BOX box;int label;float confidence;
};bool cmp(Object &obj1, Object &obj2){return obj1.confidence > obj2.confidence;
}vector<list<Object>> NMS(std::vector<Object> objs, float iou_thres = 0.45){//第一步:将所有矩形框按照不同的类别标签分组,组内按照置信度高低得分进行排序;list<Object> obj_l;vector<list<Object>> NMS_List;int a = 0;for(int i = 0; i < 1; i++){for(auto j : objs){if(j.label == i){obj_l.push_back(j);obj_l.sort(cmp);        //依据置信度升序排序a = 1;}}if(a == 1){NMS_List.push_back(std::move(obj_l));a = 0;}}//第二步:计算IOUfloat x1, y1, x1_w, y1_h,x2, y2, x2_w, y2_h;float x_box, y_box, x_w_box, y_h_box, w_box, h_box;float S1,S2,SBOX,res_iou;int row = NMS_List.size();  //行数     列数:NMS_List[0].size()int tmp;for(int i = 0; i < row ; i++)  //不同分类的循环{tmp = 0;list<Object>::iterator it = NMS_List[i].begin();while(it != --NMS_List[i].end()){x1 = it->box.x;y1 = it->box.y;x1_w = x1 + it->box.width;y1_h = y1 + it->box.height;while(it != --NMS_List[i].end()){it++;x2 = it->box.x;y2 = it->box.y;x2_w = x2 + it->box.width;y2_h = y2 + it->box.height;//交集左上角坐标x_box,y_box  框1-x1和框2-x2的最大值   框1-y1和框2-y2的最大值x_box = std::max(x1, x2);y_box = std::max(y1, y2);//交集右下角坐标x_w_box,y_h_box  框1-x1_w和框2-x2_w的最小值  框1-y1_h和框2-y2_h的最小值x_w_box = std::min(x1_w, x2_w);y_h_box = std::min(y1_h, y2_h);//交集框宽高w_box = x_w_box - x_box;h_box = y_h_box - y_box;//无交集情况if(w_box <= 0 || h_box <= 0){it = NMS_List[i].erase(it);if(it == NMS_List[i].end()){break;}it--;continue;}//有交集,计算IOUS1 = (x1_w - x1) * (y1_h - y1);S2 = (x2_w - x2) * (y2_h - y2);SBOX = w_box * h_box;if((res_iou = SBOX / (S1 + S2 - SBOX)) > iou_thres){it = NMS_List[i].erase(it);if(it == NMS_List[i].end()){break;}it--;}}it = NMS_List[i].begin();if(it == --NMS_List[i].end()){break;}tmp++;for(int z = 0; z < tmp; z++){it++;if(it == --NMS_List[i].end()){break;}}}}return NMS_List;
}

main.cpp

#include<fstream>  
#include<iostream>  
#include<opencv2/opencv.hpp>
#include <opencv2/core/core.hpp>
#include <opencv2/imgproc.hpp>
#include <iomanip>  //保留小数#include "NvInfer.h"
#include "logging.h"#include "infer.h"using namespace std;
using namespace nvinfer1;class MyLogger : public nvinfer1::ILogger {public:explicit MyLogger(nvinfer1::ILogger::Severity severity =nvinfer1::ILogger::Severity::kWARNING): severity_(severity) {}void log(nvinfer1::ILogger::Severity severity,const char *msg) noexcept override {if (severity <= severity_) {std::cerr << msg << std::endl;}}nvinfer1::ILogger::Severity severity_;
};int main()
{//一、图像处理const int model_width = 640;const int model_height = 640;string image_path = "../data/12.png";  //填写自己图片路径float* input_blob = new float[model_height * model_width * 3];cv::Mat input_image = cv::imread(image_path);cv::Mat resize_image;const float ratio = std::min(model_width / (input_image.cols * 1.0f),model_height / (input_image.rows * 1.0f));// 等比例缩放const int border_width = input_image.cols * ratio;const int border_height = input_image.rows * ratio;// 计算偏移值const int x_offset = (model_width - border_width) / 2;const int y_offset = (model_height - border_height) / 2;//将输入图像缩放至resize_imagecv::resize(input_image, resize_image, cv::Size(border_width, border_height));//复制图像并且制作边界cv::copyMakeBorder(resize_image, resize_image, y_offset, y_offset, x_offset,x_offset, cv::BORDER_CONSTANT, cv::Scalar(114, 114, 114));// 转换为RGB格式cv::cvtColor(resize_image, resize_image, cv::COLOR_BGR2RGB);// cv::imshow("12", resize_image);// cv::waitKey(0);//归一化const int channels = resize_image.channels();const int width = resize_image.cols;const int height = resize_image.rows;for (int c = 0; c < channels; c++) {for (int h = 0; h < height; h++) {for (int w = 0; w < width; w++) {input_blob[c * width * height + h * width + w] =resize_image.at<cv::Vec3b>(h, w)[c] / 255.0f;  //at<Vec3b> 是 OpenCV 中用于访问图像像素的一种方法,使用 at<Vec3b> 获取彩色图像中特定位置的像素颜色值}}}//二、模型反序列化MyLogger logger;//读取trt信息const std::string engine_file_path = "../data/best.engine";  //填写自己trt文件路径std::stringstream engine_file_stream;engine_file_stream.seekg(0, engine_file_stream.beg);  //从起始位置偏移0个字节,指针移动到文件流的开头std::ifstream ifs(engine_file_path);engine_file_stream << ifs.rdbuf();ifs.close();engine_file_stream.seekg(0, std::ios::end);         //先把文件输入流指针定位到文档末尾来获取文档的长度const int model_size = engine_file_stream.tellg();  //获取文件流的总长度engine_file_stream.seekg(0, std::ios::beg);void *model_mem = malloc(model_size);               //开辟一样长的空间engine_file_stream.read(static_cast<char *>(model_mem), model_size);    //将内容读取到model_mem中nvinfer1::IRuntime *runtime = nvinfer1::createInferRuntime(logger);nvinfer1::ICudaEngine *engine = runtime->deserializeCudaEngine(model_mem, model_size);free(model_mem);//三、模型推理nvinfer1::IExecutionContext *context = engine->createExecutionContext();void *buffers[2];// 获取模型输入尺寸并分配GPU内存nvinfer1::Dims input_dim = engine->getBindingDimensions(0);int input_size = 1;for (int j = 0; j < input_dim.nbDims; ++j) {input_size *= input_dim.d[j];}cudaMalloc(&buffers[0], input_size * sizeof(float));// 获取模型输出尺寸并分配GPU内存nvinfer1::Dims output_dim = engine->getBindingDimensions(1);int output_size = 1;for (int j = 0; j < output_dim.nbDims; ++j) {output_size *= output_dim.d[j];}cudaMalloc(&buffers[1], output_size * sizeof(float));// 给模型输出数据分配相应的CPU内存float *output_buffer = new float[output_size];//cudaStream_t stream;cudaStreamCreate(&stream);// 拷贝输入数据cudaMemcpyAsync(buffers[0], input_blob,input_size * sizeof(float),cudaMemcpyHostToDevice, stream);// 投入数据流、执行推理if(context->enqueueV2(buffers, stream, nullptr)){cout << "enqueueV2执行推理成功" << endl;}else{cout << "enqueueV2执行推理失败" << endl;return -1;}// 拷贝输出数据cudaMemcpyAsync(output_buffer, buffers[1],output_size * sizeof(float),cudaMemcpyDeviceToHost, stream);cudaStreamSynchronize(stream);delete context;delete engine;delete runtime;delete[] input_blob;//四、输出结果output_buffer,放入objs  xywh为中心点坐标 和宽高int nc = 1;     // 自己的识别目标种类数float *ptr = output_buffer;std::vector<Object> objs;for (int i = 0; i < 25200; ++i) {const float objectness = ptr[4];if (objectness >= 0.45f) {const int label = std::max_element(ptr + 5, ptr + 5 + nc) - (ptr + 5);  //std::max_element返回范围内的最大元素const float confidence = ptr[5 + label] * objectness;if (confidence >= 0.25f) {const float bx = ptr[0];const float by = ptr[1];const float bw = ptr[2];const float bh = ptr[3];// std::cout << bx << "," << by << "," << bw << "," << bh << std::endl;Object obj;// 还原图像尺寸中box的尺寸比例,这里要减掉偏移值,并把box中心点坐标xy转成左上角坐标xyobj.box.x = (bx - bw * 0.5f - x_offset) / ratio;obj.box.y = (by - bh * 0.5f - y_offset) / ratio;obj.box.width = bw / ratio;obj.box.height = bh / ratio;obj.label = label;obj.confidence = confidence;// std::cout << obj.box.x << "," << obj.box.y << "," << obj.box.width << "," << obj.box.height// << "," << obj.label << "," << obj.confidence << std::endl;objs.push_back(std::move(obj));}}ptr += (5+nc);}  // i loop//五、NMS非极大值抑制vector<list<Object>> finalll = NMS(objs);//六、画框int row = finalll.size();for(int i = 0; i < row; i++){list<Object>::iterator it = finalll[i].begin();while(it != finalll[i].end()){cv::Point topLeft(it->box.x, it->box.y);cv::Point bottomRight(it->box.x + it->box.width, it->box.y + it->box.height);cv::rectangle(input_image, topLeft, bottomRight, cv::Scalar(0, 0, 255), 2);std::stringstream buff;buff.precision(2);  //覆盖默认精度,置信度保留2位小数buff.setf(std::ios::fixed);buff << it->confidence;string text =names[it->label] + " " + buff.str();cv::putText(input_image, text, topLeft, 0, 1, cv::Scalar(0, 255, 0), 2);it++;}}cv::imwrite("pig.jpg", input_image);return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Sublime Text】| 01——下载安装注册

系列文章目录 【Sublime Text】| 01——下载软件安装并注册 【Sublime Text】| 02——常用插件安装及配置 失败了也挺可爱&#xff0c;成功了就超帅。 文章目录 前言1. 下载2. 安装3. 注册3.1 通过修改应用程序注册3.2 通过替换应用程序注册 感谢 前言 轻量代码编辑器有很多 之…

国家发改委:《电能质量管理办法(暂行)》2024年4月1日起施行

中华人民共和国国家发展和改革委员会令 第8号 《电能质量管理办法(暂行)》已经2023年12月26日第7次委务会议审议通过,现予公布,自2024 年4月1日起施行。 主任 郑栅洁 2023年12月27日 电能质量管理办法&#xff08;暂行&#xff09; 第一章 总则 第一条 为加强电能质量管理&…

中小型企业如何在当今的商业环境中占得一席之地?

在电商新零售的浪潮下&#xff0c;一种全新的消费模式正在崭露头角——消费增值模式。这种模式不仅改变了消费者的购物体验&#xff0c;更在电商领域掀起了一场革命。本文将通过具体数据&#xff0c;为您揭示消费增值模式的魅力。 一、消费增值模式的奥秘 消费增值模式的核心在…

【DevOps-08-2】Harbor的基本操作

一、简要描述 Harbor作为镜像仓库,主要的交互方式就是将镜像上传到Harbor上,以及从Harbor上下载指定镜像 在传输镜像前,可以先使用Harbor提供的权限管理,将项目设置为私有项目,并对不同用户设置不同角色,从而更方便管理镜像。 二、Harbor添加用户和项目 1、添加Harbor用…

一篇文告诉你:到底该如何搭建好企业知识库

全球化的竞争环境和瞬息万变的市场使得企业需要一种管理和整合各种信息与知识的方式。这就是企业知识库&#xff0c;一种核心资源&#xff0c;能够帮助企业提升工作效能&#xff0c;并更好地支持决策、创新和学习。但是&#xff0c;如何优化知识库&#xff0c;让它成为企业的有…

迁移学习的最新进展和挑战

随着深度学习和人工智能技术的飞速发展&#xff0c;迁移学习作为一种有效的机器学习方法&#xff0c;已经在各个领域取得了显著的成果。迁移学习是指将一个领域&#xff08;源领域&#xff09;的知识应用到另一个领域&#xff08;目标领域&#xff09;&#xff0c;以提高目标领…

spark基础--学习笔记

1 spark 介绍 1.1 spark概念 Apache Spark是专为大规模数据处理而设计的快速通用的分布式计算引擎&#xff0c;是开源的类Hadoop MapReduce的通用分布式计算框架。和MapReduce一样&#xff0c;都是完成大规模数据的计算处理。 简而言之&#xff0c;Spark 借鉴了 MapReduce思…

在1G的内存中,对百亿个QQ号去重?

文章目录 一、公共方法1、生成模拟QQ号2、读取数据文件3、测试方法 二、HashSet三、Java8的Stream四、Segment五、BloomFilter六、BitMap七、总结 假设QQ号是int类型&#xff0c;那么最多可以有4294967295个&#xff0c;就是43亿左右&#xff0c;QQ号无论多少位&#xff0c;每个…

【链表】力扣206反转链表

题目 力扣206反转链表 思路图解 代码实现 双指针代码实现 public static ListNode reverseList(ListNode head) {// 初始化pre&#xff0c;curListNode pre null;ListNode cur head;// 当cur为null时&#xff0c;说明反转结束while(cur ! null) {// 临时保存cur.next节点…

Python写冒泡

当你要用Python写冒泡排序算法时&#xff0c;你可以使用下面的代码&#xff1a; def bubble_sort(arr):n len(arr)for i in range(n-1):for j in range(n-i-1):if arr[j] > arr[j1]:arr[j], arr[j1] arr[j1], arr[j]return arr这个函数接受一个数组作为输入&#xff0c;并…

linux开发板静态IP无法ping通外网

硬件平台&#xff1a;韦东山的6ull开发板 问题&#xff1a; 使用网线直连路由器&#xff0c;动态获取IP时能ping通外网&#xff1b; 改为静态IP时&#xff0c;能ping通局域网&#xff0c;但无法ping通外网。 改为静态IP&#xff1a;修改/etc/network/interfaces 测试&#…

CentOS本地部署SQL Server数据库无公网ip环境实现远程访问

文章目录 前言1.安装GeoServer2. windows 安装 cpolar3. 创建公网访问地址4. 公网访问Geo Servcer服务5. 固定公网HTTP地址 前言 GeoServer是OGC Web服务器规范的J2EE实现&#xff0c;利用GeoServer可以方便地发布地图数据&#xff0c;允许用户对要素数据进行更新、删除、插入…

Linux系统——测试端口连通性方法

目录 一、TCP端口连通性测试 1、ssh 2、telnet&#xff08;可能需要安装&#xff09; 3、curl 4、tcping&#xff08;需要安装&#xff09; 5、nc&#xff08;需要安装&#xff09; 6、nmap&#xff08;需要安装&#xff09; 二、UDP端口连通性测试 1、nc&#xff08;…

adb forward使用

adb forward是Android Debug Bridge&#xff08;ADB&#xff09;的一个命令&#xff0c;它可以将设备端口和主机端口之间建立一个转发通道&#xff0c;从而使主机可以通过该通道访问设备端口提供的服务。使用adb forward可以方便地进行端口转发&#xff0c;例如在电脑上运行的应…

spring-boot-admin-server-ui 打包备忘

spring-boot-admin-server-ui 打包备忘 先试一下springboot2.0*&#xff0c;这是一个老项目 ui包里面发现 "node-sass": "^4.11.0",Node.js 版本node-sass 版本16.x6.x15.x5.x14.x4.14.x13.x4.13.x12.x4.12.x11.x4.10.x10.x4.9.x8.x4.5.3 先把node调成1…

【MySQL】MySQL版本8+ 的 with recursive 两种递归语法的使用

力扣题 1、题目地址 1270. 向公司 CEO 汇报工作的所有人 2、模拟表 员工表&#xff1a;Employees Column NameTypeemployee_idintemployee_namevarcharmanager_idint employee_id 是这个表具有唯一值的列。这个表中每一行中&#xff0c;employee_id 表示职工的 ID&#x…

从0到1实战微服务架构之Nacos下载安装

目录 一、前言 二、Nacos概述 三、Nacos架构 3.1 Open API 3.2 Config Service 3.3 Naming Service 3.4 Nacos Core 3.5 Consistency Protocol 四、Nacos部署实践 4.1 Nacos下载 4.2 Nacos部署 五、总结 一、前言 Nacos是一个开源的、易于使用的、功能丰富的平台&a…

19道ElasticSearch面试题(很全)

点击下载《19道ElasticSearch面试题&#xff08;很全&#xff09;》 1. elasticsearch的一些调优手段 1、设计阶段调优 &#xff08;1&#xff09;根据业务增量需求&#xff0c;采取基于日期模板创建索引&#xff0c;通过 roll over API 滚动索引&#xff1b; &#xff08;…

谓词-量词、主析取、主和取范式、前束范式、推理证明

这部分内容&#xff0c;主要需要掌握谓词推理&#xff0c;而前提是掌握将自然语言符号化为谓词、用量词来限定辖域&#xff0c;量词的消去、剩下就是推理过程。还需要掌握的是主析取、主和取范式和前束范式。 存在量词∃&#xff1a;至少有一个 全称量词∀&#xff1a;全都是…

Linux驱动学习—输入子系统

1、什么是输入子系统&#xff1f; 输入子系统是Linux专门做的一套框架来处理输入事件的&#xff0c;像鼠标&#xff0c;键盘&#xff0c;触摸屏这些都是输入设备&#xff0c;但是这邪恶输入设备的类型又都不是一样的&#xff0c;所以为了统一这些输入设备驱动标准应运而生的。…