在项目程序中经常看到动态链接库,非常好奇,想自己实现一下,于是乎尝试一波。就因为这种好奇,每天都被bug所困扰。。。

1. 训练caffemodel

在windows环境下搭建caffe无果,转投Ubuntu。。。

用的caffe--example--mnist中的文件,新建文件夹的话注意改路径,下面为train.sh

#!/usr/bin/env sh
set -e /home/fish/caffe/build/tools/caffe train --solver=/home/fish/STUDY/lenet_solver.prototxt

训练好后把lenet_train_test.prototxt和训练好的模型lenet_iter_10000.caffemodel拿出来。

2. 使用cv::dnn里的API加载model,输入图片,进行测试(可跳过)

根据文章https://blog.csdn.net/sushiqian/article/details/78555891,修改模型文件。若图片为白底黑字,bitwise_not一下。

#include
#include <opencv2/opencv.hpp>
#include <opencv2/dnn.hpp> using namespace std;
using namespace cv;
using namespace cv::dnn; /* Find best class for the blob (i. e. class with maximal probability) */
static void getMaxClass(const Mat& probBlob, int* classId, double* classProb)
{
Mat probMat = probBlob.reshape(1, 1);
Point classNumber;
minMaxLoc(probMat, NULL, classProb, NULL, &classNumber);
*classId = classNumber.x;
} int main(int argc, char* argv[])
{
string modelTxt = "C:\\Users\\ATWER\\Desktop\\lenet_train_test.prototxt";
string modelBin = "C:\\Users\\ATWER\\Desktop\\lenet_iter_10000.caffemodel";
string imgFileName = "C:\\Users\\ATWER\\Desktop\\9.png";
//read image
Mat imgSrc = imread(imgFileName);
if (imgSrc.empty()) {
cout << "Failed to read image " << imgFileName << endl;
exit(-1);
}
Mat img;
cvtColor(imgSrc, img, COLOR_BGR2GRAY);
//LeNet accepts 28*28 gray image
resize(img, img, Size(28, 28));
bitwise_not(img, img);
img /= 255; //transfer image(1*28*28) to blob data with 4 dimensions(1*1*28*28)
Mat inputBlob = dnn::blobFromImage(img);
dnn::Net net;
try {
net = dnn::readNetFromCaffe(modelTxt, modelBin);
}
catch (cv::Exception& ee) {
cerr << "Exception: " << ee.what() << endl;
if (net.empty()) {
cout << "Can't load the network by using the flowing files:" << endl;
cout << "modelTxt: " << modelTxt << endl;
cout << "modelBin: " << modelBin << endl; exit(-1);
}
}
Mat pred;
net.setInput(inputBlob, "data");//set the network input, "data" is the name of the input layer
pred = net.forward("prob");//compute output, "prob" is the name of the output layer
cout << pred << endl; int classId; double classProb; getMaxClass(pred, &classId, &classProb);
cout << "Best Class: " << classId << endl;
cout << "Probability: " << classProb * 100 << "%" << endl;
}

3. 创建动态链接库

参考https://blog.csdn.net/qq_30139555/article/details/103621955

class.h

#include
#include <opencv2/opencv.hpp>
#include <opencv2/dnn/dnn.hpp> using namespace std;
using namespace cv;
using namespace cv::dnn; extern "C" _declspec(dllexport) void Classfication(char* imgpath, char* result);

在此处卡的最久,原本我写的是Classfication(string imgpath, string result),生成dll时没问题,调用时总是System.AccessViolationException: 尝试读取或写入受保护的内存。后来发现要写成指针的形式。

class.cpp

#include
#include <opencv2/opencv.hpp>
#include <opencv2/dnn/dnn.hpp>
#include "class.h"
using namespace std;
using namespace cv;
using namespace cv::dnn; /* Find best class for the blob (i. e. class with maximal probability) */
static void getMaxClass(const Mat& probBlob, int* classId, double* classProb)
{
Mat probMat = probBlob.reshape(1, 1);
Point classNumber;
minMaxLoc(probMat, NULL, classProb, NULL, &classNumber);
*classId = classNumber.x;
} void Classfication(char* imgpath, char* result)
{
string res = "";
string modelTxt = "C:\\Users\\ATWER\\Desktop\\lenet_train_test.prototxt";
string modelBin = "C:\\Users\\ATWER\\Desktop\\lenet_iter_10000.caffemodel";
//string imgFileName = "C:\\Users\\ATWER\\Desktop\\9.png";
string imgFileName = imgpath;
//read image
Mat imgSrc = imread(imgFileName);
if (imgSrc.empty()) {
cout << "Failed to read image " << imgFileName << endl;
exit(-1);
}
Mat img;
cvtColor(imgSrc, img, COLOR_BGR2GRAY);
//LeNet accepts 28*28 gray image
resize(img, img, Size(28, 28));
bitwise_not(img, img);
img /= 255; //transfer image(1*28*28) to blob data with 4 dimensions(1*1*28*28)
Mat inputBlob = dnn::blobFromImage(img);
dnn::Net net;
try {
net = dnn::readNetFromCaffe(modelTxt, modelBin);
}
catch (cv::Exception& ee) {
cerr << "Exception: " << ee.what() << endl;
if (net.empty()) {
cout << "Can't load the network by using the flowing files:" << endl;
cout << "modelTxt: " << modelTxt << endl;
cout << "modelBin: " << modelBin << endl; exit(-1);
}
}
Mat pred;
net.setInput(inputBlob, "data");//set the network input, "data" is the name of the input layer
pred = net.forward("prob");//compute output, "prob" is the name of the output layer
int classId;
   double classProb;
   getMaxClass(pred, &classId, &classProb);
res += to_string(classId);
res += '|';
res += to_string(classProb);
strcpy_s(result, 15, res.c_str());
}

4. 调用动态链接库

根据数据的长度申请非托管空间参考:https://blog.csdn.net/xiaoyong_net/article/details/50178021

文中说:“一定要加1,否则后面是乱码,原因未找到 ”,应该是打印字符串时会打印到“\n”为止,没有遇到\n会一直打印下去。.Length方法没有计算"\n",+1的空间用于存放“\n”。

using System;
using System.Runtime.InteropServices; namespace Test
{
class Program
{
[DllImport("E:/c++project/caffedll/x64/Debug/caffedll.dll", EntryPoint = "Classfication")] unsafe private static extern void Classfication(IntPtr imgpath, IntPtr result);
private static IntPtr mallocIntptr(string strData)
{
//先将字符串转化成字节方式
Byte[] btData = System.Text.Encoding.Default.GetBytes(strData);
//申请非拖管空间
IntPtr m_ptr = Marshal.AllocHGlobal(btData.Length);
//给非拖管空间清0
Byte[] btZero = new Byte[btData.Length + 1]; //一定要加1,否则后面是乱码,原因未找到
Marshal.Copy(btZero, 0, m_ptr, btZero.Length);
//给指针指向的空间赋值
Marshal.Copy(btData, 0, m_ptr, btData.Length);
return m_ptr;
}
private static IntPtr mallocIntptr(int length)
{
//申请非拖管空间
IntPtr m_ptr = Marshal.AllocHGlobal(length);
//给非拖管空间清0
Byte[] btZero = new Byte[length];
Marshal.Copy(btZero, 0, m_ptr, btZero.Length);
//给指针指向的空间赋值
Marshal.Copy(btZero, 0, m_ptr, length);
return m_ptr;
}
static void Main(string[] args)
{
string s = "C:\\Users\\ATWER\\Desktop\\9.png";
IntPtr ptrFileName;
IntPtr res;
//根据数据的长度申请非托管空间
ptrFileName = mallocIntptr(s);
res = mallocIntptr(50);
Classfication(ptrFileName, res);
string result = Marshal.PtrToStringAnsi(res);
string[] a = result.Split('|');
Console.WriteLine("class:"+a[0]+"\n"+"score:"+a[1]);
Marshal.FreeHGlobal(res);
}
}
}

将mnist训练的caffemodel生成动态链接库DLL的更多相关文章

  1. MinGW gcc 生成动态链接库 dll 的一些问题汇总 (补充)

    我以前写过一个小短文,介绍MinGW gcc 生成动态链接库 dll 的一些问题.当时写的并不全面.近期又遇到写新的问题.这里记录一下,做个补充. 通常情况下,dll 中的函数假设採用 _stdcal ...

  2. dll = MinGW gcc 生成动态链接库 dll 的一些问题汇总

    MinGW gcc 生成动态链接库 dll 的一些问题汇总 https://blog.csdn.net/liyuanbhu/article/details/42612365 网络上关于用 MinGW  ...

  3. 使用python创建生成动态链接库dll

    如今,随着深度学习的发展,python已经成为了深度学习研究中第一语言.绝大部分的深度学习工具包都有python的版本,很多重要算法都有python版本的实现.为了将这些算法应用到具体工程中,这些工具 ...

  4. MinGW gcc 生成动态链接库 dll 的一些问题汇总(由浅入深,很详细)

    网络上关于用 MinGW gcc 生成动态链接库的文章很多.介绍的方法也都略有不同.这次我在一个项目上刚好需要用到,所以就花了点时间将网上介绍的各种方法都实验了一遍.另外,还根据自己的理解试验了些网上 ...

  5. 从一到二:利用mnist训练集生成的caffemodel对mnist测试集与自己手写的数字进行测试

    通过从零到一的教程,我们已经得到了通过mnist训练集生成的caffemodel,主要包含下面四个文件: 接下来就可以利用模型进行测试了.关于测试方法按照上篇教程还是选择bat文件,当然python. ...

  6. 利用mnist训练集生成的caffemodel对mnist测试集与自己手写的数字进行测试

    从一到二:利用mnist训练集生成的caffemodel对mnist测试集与自己手写的数字进行测试 通过从零到一的教程,我们已经得到了通过mnist训练集生成的caffemodel,主要包含下面四个文 ...

  7. 动态链接库DLL的创建生成及调用

    一.背景 最近在做CANTOUSB底层驱动的调用,是调用别人已经封装好的库,看不到别人写的源程序.程序中调用的是隐式调用即 x.h+x.lib+x.dll,其中DLL即是动态链接库(Dynamic L ...

  8. C#调用C++生成的动态链接库DLL

    一.背景 由于要使用C#写app,所以要把C++生成的DLL在C#中调用,所以就涉及怎样去调用外部的dll问题. 二.C#调用外部DLL 首先先看下C#调用外部DLL的代码 using System. ...

  9. 编译可供C#调用的C/C++动态链接库dll文件

    编译可供C#调用的C/C++动态链接库dll文件,C语言控制台应用程序,探索生成dll过程 由于项目需求,需要公司另一个团队提供相关算法支持,是用C语言编译好的dll库提供给我们进行调用. 但是拿到d ...

  10. VC++动态链接库(DLL)编程深入浅出(zz)

    VC++动态链接库(DLL)编程深入浅出(zz) 1.概论 先来阐述一下DLL(Dynamic Linkable Library)的概念,你可以简单的把DLL看成一种仓库,它提供给你一些可以直接拿来用 ...

随机推荐

  1. ActiveMQ c# 系列——实例(二)

    前言 前面一章中介绍了activemq,并且呢安装了. 这一章就来看一下实例吧. 正文 我使用队列举例. 是这样子,队列是一对一的关系,比如说我生产了一条消息,那么只要有一个消费者消费完毕那么就算消费 ...

  2. 4A 安全之授权:编程的门禁,你能解开吗?

    概述 在安全管理系统里面,授权(Authorization)的概念常常是和认证(Authentication).账号(Account)和审计(Audit)一起出现的,并称之为 4A.就像上一文章提到的 ...

  3. Springmvc来做甘特图的显示控制。

    springmvc没有Springboot好用:所以就用了ajax通信来获取数据,这其中有不少坑,我来带大家踩一踩. 1.在控制层中,接口不能直接返回Json格式. 我的解决方法可以使通过map,或者 ...

  4. three.js实现数字孪生3D仓库一期(开源)

    大家好,本文使用three.js实现了3D仓库一期项目,给出了代码,分析了关键点,感谢大家~ 关键词:数字孪生.three.js.Web3D.WebGL.智慧仓库.开源 代码:Github 我正在承接 ...

  5. 力扣1083(MySQL)-销售分析Ⅲ(简单)

    题目: Table: Product Table: Sales 编写一个SQL查询,报告2019年春季才售出的产品.即仅在2019-01-01至2019-03-31(含)之间出售的商品. 以 任意顺序 ...

  6. 力扣461(java)-汉明距离(简单)

    题目: 两个整数之间的 汉明距离 指的是这两个数字对应二进制位不同的位置的数目. 给你两个整数 x 和 y,计算并返回它们之间的汉明距离. 示例 1: 输入:x = 1, y = 4输出:2解释:1 ...

  7. Docker部署Node应用简单实践

    简介: 本文将从零至一,介绍如何在云服务器上通过 Docker 容器运行一个简单的Node应用. 前言 本文将从零至一,介绍如何在云服务器上通过 Docker 容器运行一个简单的Node应用.本文假设 ...

  8. ModelScope初探:一行代码调用成熟AI模型。

    简介: 如何用一行代码调用成熟AI模型?试试ModelScope,让AI开发者解放生产力! ModelScope是阿里推出的下一代开源的模型即服务共享平台,为泛AI开发者提供灵活.易用.低成本的一站式 ...

  9. KubeVela 1.3 发布:开箱即用的可视化应用交付平台,引入插件生态、权限认证、版本化等企业级新特性

    ​简介:得益于 KubeVela 社区上百位开发者的参与和 30 多位核心贡献者的 500 多次代码提交, KubeVela 1.3 版本正式发布.相较于三个月前发布的 v1.2 版本[1],新版本在 ...

  10. 压测场景下的 TIME_WAIT 处理

    简介: 压测场景下的 TIME_WAIT 处理 1. 序 某专有云项目具备压测场景,在Windows的压测机上用 LoadRunner 进行业务的压力测试,压测运行一段时间后出现大量端口无法分配的报错 ...