MNN学习笔记(六):配置visual studio项目

2023-12-23 10:48

本文主要是介绍MNN学习笔记(六):配置visual studio项目,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这个其实很简单,原因是MNN项目组已经提供了编译好的库:

1.下载编译好的MNN库

下载地址为:https://github.com/alibaba/MNN/releases

下载两个文件:Source.code和MNN-WindowsX64-0.2.1.7.zip

2.在visual studio上进行配置

注意visual studio版本为2017,我习惯把这些库都跟opencv放一起,具体来讲就是:

首先,把从Source code中解压的include文件如图1所示,复制出来;

然后,在自己的opencv的include文件夹下面新建一个MNN子文件夹,将上面的头文件复制过去:

其次,将解压好的MNN-WindowsX64-0.2.1.7.zip文件中MNN.dll和MNN.lib放到opencv对应位置:

最后,跟配置opencv一样,新建一个项目,配置项目:

3.测试代码

总共三个文件:

第一个文件:mobilenetssd.h

#ifndef _MOBILENET_SSD_H_
#define _MOBILENET_SSD_H_#include <vector>#include "MNN/Interpreter.hpp"
#include "MNN/MNNDefine.h"
#include "MNN/Tensor.hpp"
#include "MNN/ImageProcess.hpp"#include "opencv2/core.hpp"
#include "opencv2/imgproc.hpp"namespace mirror {
struct ObjectInfo {std::string name_;cv::Rect location_;float score_;
};class MobilenetSSD {
public:MobilenetSSD();~MobilenetSSD();int Init(const char* root_path);int Detect(const cv::Mat& img_src, std::vector<ObjectInfo>* objects);
private:uint8_t* GetImage(const cv::Mat& img_src) {uchar* data_ptr = new uchar[img_src.total() * 4];cv::Mat img_tmp(img_src.size(), CV_8UC4, data_ptr);cv::cvtColor(img_src, img_tmp, CV_BGR2RGBA, 4);return (uint8_t*)img_tmp.data;}private:bool initialized_;const cv::Size inputSize_ = { 300, 300 };std::vector<int> dims_ = { 1, 3, 300, 300 };const float meanVals_[3] = { 0.5f, 0.5f, 0.5f };const float normVals_[3] = { 0.007843f, 0.007843f, 0.007843f };std::vector<std::string> class_names = {"background", "aeroplane", "bicycle", "bird", "boat","bottle", "bus", "car", "cat", "chair","cow", "diningtable", "dog", "horse","motorbike", "person", "pottedplant","sheep", "sofa", "train", "tvmonitor"};std::shared_ptr<MNN::Interpreter> mobilenetssd_interpreter_;MNN::Session* mobilenetssd_sess_ = nullptr;MNN::Tensor* input_tensor_ = nullptr;std::shared_ptr<MNN::CV::ImageProcess> pretreat_data_ = nullptr;};}#endif // !_MOBILENET_SSD_H_

第二个文件:mobilenetssd.cpp

#include "mobilenetssd.h"
#include <iostream>
#include <string>#include "opencv2/imgproc.hpp"namespace mirror {MobilenetSSD::MobilenetSSD() {initialized_ = false;
}MobilenetSSD::~MobilenetSSD() {mobilenetssd_interpreter_->releaseModel();mobilenetssd_interpreter_->releaseSession(mobilenetssd_sess_);
}int MobilenetSSD::Init(const char * root_path) {std::cout << "start Init." << std::endl;std::string model_file = std::string(root_path) + "/mobilenetssd.mnn";mobilenetssd_interpreter_ = std::unique_ptr<MNN::Interpreter>(MNN::Interpreter::createFromFile(model_file.c_str()));if (nullptr == mobilenetssd_interpreter_) {std::cout << "load model failed." << std::endl;return 10000;}MNN::ScheduleConfig schedule_config;schedule_config.type = MNN_FORWARD_CPU;schedule_config.numThread = 4;MNN::BackendConfig backend_config;backend_config.precision = MNN::BackendConfig::Precision_High;backend_config.power = MNN::BackendConfig::Power_High;schedule_config.backendConfig = &backend_config;mobilenetssd_sess_ = mobilenetssd_interpreter_->createSession(schedule_config);// image processerMNN::CV::Matrix trans;trans.setScale(1.0f, 1.0f);MNN::CV::ImageProcess::Config img_config;img_config.filterType = MNN::CV::BICUBIC;::memcpy(img_config.mean, meanVals_, sizeof(meanVals_));::memcpy(img_config.normal, normVals_, sizeof(normVals_));img_config.sourceFormat = MNN::CV::RGBA;img_config.destFormat = MNN::CV::RGB;pretreat_data_ = std::shared_ptr<MNN::CV::ImageProcess>(MNN::CV::ImageProcess::create(img_config));pretreat_data_->setMatrix(trans);std::string input_name = "data";input_tensor_ = mobilenetssd_interpreter_->getSessionInput(mobilenetssd_sess_, input_name.c_str());mobilenetssd_interpreter_->resizeTensor(input_tensor_, dims_);mobilenetssd_interpreter_->resizeSession(mobilenetssd_sess_);initialized_ = true;std::cout << "end Init." << std::endl;return 0;
}int MobilenetSSD::Detect(const cv::Mat & img_src, std::vector<ObjectInfo>* objects) {std::cout << "start detect." << std::endl;if (!initialized_) {std::cout << "model uninitialized." << std::endl;return 10000;}if (img_src.empty()) {std::cout << "input empty." << std::endl;return 10001;}int width = img_src.cols;int height = img_src.rows;// preprocesscv::Mat img_resized;cv::resize(img_src, img_resized, inputSize_);uint8_t* data_ptr = GetImage(img_resized);pretreat_data_->convert(data_ptr, inputSize_.width, inputSize_.height, 0, input_tensor_);mobilenetssd_interpreter_->runSession(mobilenetssd_sess_);std::string output_name = "detection_out";MNN::Tensor* output_tensor = mobilenetssd_interpreter_->getSessionOutput(mobilenetssd_sess_, output_name.c_str());// copy to hostMNN::Tensor output_host(output_tensor, output_tensor->getDimensionType());output_tensor->copyToHostTensor(&output_host);auto output_ptr = output_host.host<float>();for (int i = 0; i < output_host.height(); ++i) {int index = i * output_host.width();ObjectInfo object;object.name_ = class_names[int(output_ptr[index + 0])];object.score_ = output_ptr[index + 1];object.location_.x = output_ptr[index + 2] * width;object.location_.y = output_ptr[index + 3] * height;object.location_.width = output_ptr[index + 4] * width - object.location_.x;object.location_.height = output_ptr[index + 5] * height - object.location_.y;objects->push_back(object);}std::cout << "end detect." << std::endl;return 0;
}}

第三个文件:main.cpp

#include "mobilenetssd.h"
#include "opencv2/opencv.hpp"int main(int argc, char* argv[]){const char* img_path = "./data/images/test.jpg";cv::Mat img_src = cv::imread(img_path);mirror::MobilenetSSD* mobilenetssd = new mirror::MobilenetSSD();const char* root_path = "./data/models";mobilenetssd->Init(root_path);std::vector<mirror::ObjectInfo> objects;mobilenetssd->Detect(img_src, &objects);int num_objects = static_cast<int>(objects.size());for (int i = 0; i < num_objects; ++i) {std::cout << "location: " << objects[i].location_ << std::endl;cv::rectangle(img_src, objects[i].location_, cv::Scalar(255, 0, 255), 2);char text[256];sprintf_s(text, "%s %.1f%%", objects[i].name_.c_str(), objects[i].score_ * 100);int baseLine = 0;cv::Size label_size = cv::getTextSize(text, cv::FONT_HERSHEY_SIMPLEX, 0.5, 1, &baseLine);cv::putText(img_src, text, cv::Point(objects[i].location_.x,objects[i].location_.y + label_size.height),cv::FONT_HERSHEY_SIMPLEX, 0.5, cv::Scalar(0, 0, 0));}cv::imwrite("./data/images/cat.jpg", img_src);cv::imshow("result", img_src);cv::waitKey(0);delete mobilenetssd;system("pause");return 0;
}

最后结果:

需要用到的模型和测试图片下载地址:https://download.csdn.net/download/sinat_31425585/12137855

打完收工!

这篇关于MNN学习笔记(六):配置visual studio项目的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/527761

相关文章

Redis Cluster模式配置

《RedisCluster模式配置》:本文主要介绍RedisCluster模式配置,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录分片 一、分片的本质与核心价值二、分片实现方案对比 ‌三、分片算法详解1. ‌范围分片(顺序分片)‌2. ‌哈希分片3. ‌虚

SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志

《SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志》在SpringBoot项目中,使用logback-spring.xml配置屏蔽特定路径的日志有两种常用方式,文中的... 目录方案一:基础配置(直接关闭目标路径日志)方案二:结合 Spring Profile 按环境屏蔽关

Maven 配置中的 <mirror>绕过 HTTP 阻断机制的方法

《Maven配置中的<mirror>绕过HTTP阻断机制的方法》:本文主要介绍Maven配置中的<mirror>绕过HTTP阻断机制的方法,本文给大家分享问题原因及解决方案,感兴趣的朋友一... 目录一、问题场景:升级 Maven 后构建失败二、解决方案:通过 <mirror> 配置覆盖默认行为1. 配置示

Springboot3+将ID转为JSON字符串的详细配置方案

《Springboot3+将ID转为JSON字符串的详细配置方案》:本文主要介绍纯后端实现Long/BigIntegerID转为JSON字符串的详细配置方案,s基于SpringBoot3+和Spr... 目录1. 添加依赖2. 全局 Jackson 配置3. 精准控制(可选)4. OpenAPI (Spri

maven私服配置全过程

《maven私服配置全过程》:本文主要介绍maven私服配置全过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录使用Nexus作为 公司maven私服maven 私服setttings配置maven项目 pom配置测试效果总结使用Nexus作为 公司maven私

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Nacos注册中心和配置中心的底层原理全面解读

《Nacos注册中心和配置中心的底层原理全面解读》:本文主要介绍Nacos注册中心和配置中心的底层原理的全面解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录临时实例和永久实例为什么 Nacos 要将服务实例分为临时实例和永久实例?1.x 版本和2.x版本的区别

springboot项目中使用JOSN解析库的方法

《springboot项目中使用JOSN解析库的方法》JSON,全程是JavaScriptObjectNotation,是一种轻量级的数据交换格式,本文给大家介绍springboot项目中使用JOSN... 目录一、jsON解析简介二、Spring Boot项目中使用JSON解析1、pom.XML文件引入依