[深度学习] ncnn安装和调用基础教程
liebian365 2024-10-25 15:37 30 浏览 0 评论
1 介绍
ncnn是腾讯开发的一个为手机端极致优化的高性能神经网络前向计算框架,无第三方依赖,跨平台,但是通常都需要protobuf和opencv。ncnn目前已在腾讯多款应用中使用,如 QQ,Qzone,微信,天天P图等。ncnn主要基于C++和caffe,ncnn项目地址见:
?https://github.com/Tencent/ncnn??
本文主要介绍ncnn在Ubuntu 18和Windows10下安装和使用。ncnn实测win10和Ubuntu18下C++调用squeezenet进行分类越比opencv dnn模块调用squeeznet快3到5倍。
ncnn常见的主流系统平台和常见硬件平台都支持,包括树莓派。
其他平台安装教程见:
??https://github.com/Tencent/ncnn/wiki/how-to-build#build-for-windows-x64-using-visual-studio-community-2017??
2 Ubuntu 18下ncnn安装和使用
2.1 Ubuntu 18下ncnn编译安装
ncnn在linux平台需要protobuf和较低的gcc版本支持(根据评论区baiyu33的回复,最新ncnn版本在gcc/g++ 7.5环境,不需要降级为4.8.5,就可以编译安装protobuf 2.6.1和ncnn,2020年7月7日更新),如果你的系统可以运行caffe,可以跳过第一第二步,直接第三步。第一第二步具体安装protobuf见:
(1)将gcc、g++降级为4.8.5版本(最新版本ncnn可以跳过该阶段)。
(2)安装protobuf
(3)编译源码
先下载源码
git clone https://github.com/Tencent/ncnn
然后进入ncnn安装即可
cd ncnn
mkdir build
cd build
cmake ..
make –j12
make install
2.2 Ubuntu 18下ncnn使用
(1) 模型更新
ncnn对caffe模型支持度比其他平台模型支持度更高。所以一般都是将caffe的模型转为ncnn格式。对于比较老的caffe模型需要将其转换为最新的caffe模型(相对来说),不过一般都不需要。以alexnet为例
alexnet 的 deploy.prototxt 可以在这里下载:
??https://github.com/BVLC/caffe/tree/master/models/bvlc_alexnet??
alexnet 的 caffemodel 可以在这里下载 :
??http://dl.caffe.berkeleyvision.org/bvlc_alexnet.caffemodel??
caffe自带了工具可以把老版本的caffe 网络和模型转换为新版(ncnn的工具只认识新版),转换方法为将你需要转换的prototxt和caffemodel放在你电脑的caffe/build/tools目录下,然后终端进入caffe/build/tools,执行命令:.
./upgrade_net_proto_text old_deploy.prototxt new_deploy.prototxt
./upgrade_net_proto_binary old.caffemodel new.caffemodel
执行完成之后你就可以在caffe/build/tools下找到你的new_deploy.prototxt和new.caffemodel文件了。注意完成之后打开你的new_deploy.prototxt文件看一下,因为一般每次只需要做一个数据样本的识别,所以如果第一个 dim 不为1,要将其设为
layer {
name: "data"
type: "Input"
top: "data"
input_param { shape: { dim: 1 dim: 3 dim: 227 dim: 227 } }
}
(2)模型转换
模型转换具体使用见:
new.caffemodel文件后,需要将prototxt转换为ncnn支持的param文件,caffemodel转换为bin文件。Ncnn中使用 caffe2ncnn 工具转换为ncnn的网络描述和模型。将上面转化的new_deploy.prototxt和new.caffemodel放到ncnn/build/tools/caffe下。然后命令行输入以下命令:
./caffe2ncnn new_deploy.prototxt new.caffemodel ncnn.param ncnn.bin
或者在以下网站转换模型:
??https://convertmodel.com/?tdsourcetag=s_pcqq_aiomsg??
(3)运用模型分类
上面提到的是alexnet,在实际本文使用的是squeezenet。ncnn调用模型的一般过程都是参考ncnn/examples/下各个示例cpp文件,选择自己的模型,然后根据该目录下的CMakeLists.txt文件修改参数。本文就不修改参数了,直接选用squeezenet.cpp使用。
然后打开ncnn根目录下的CMakeLists.txt文件,将编译examples语句的注释打开(默认是被注释掉的),如图:
终端进入ncnn/build后执行:
make
然后把所获得ncnn参数和模型文件复制到ncnn/build/examples目录下,本文所用的squeezenet的ncnn文件在ncnn/examples中有提供。将模型.param和.bin文件复制到ncnn/build/examples目录下,然后终端cd到ncnn/build/examples,执行:
./squeezenet imagepath
Imagepath为图像路径。就可以得到结果。
具体调用代码直接参考ncnn/examples中的示例cpp文件,但是使用时最好使用examples所提到的模型。
(4)ncnn工程使用
如果想建立工程,需要自己新建一个目录然后编写你自己的cpp文件。比如使用ncnn/examples/squeezenet.cpp,将其放入本机任意路径新的文件夹。对于ncnn编译,需要添加opencv和openmp,编写如下CMakeLists.txt进行编译,然后cmake .就可以在当前目录运行生成的可执行文件ncnnTest。
# 设置cmake版本
cmake_minimum_required(VERSION 3.2)
# ncnn工程
project(ncnnTest)
# 调用opencv
find_package(OpenCV REQUIRED)
# 调用openmp
FIND_PACKAGE( OpenMP REQUIRED)
if(OPENMP_FOUND)
message("OPENMP FOUND")
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${OpenMP_C_FLAGS}")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${OpenMP_CXX_FLAGS}")
set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} ${OpenMP_EXE_LINKER_FLAGS}")
endif()
# 包含ncnn的头文件
include_directories(/home/user/ncnn/build/install/include/ncnn)
# 包含ncnn的链接文件
link_directories(/home/user/ncnn/build/install/lib)
# 生成可执行文件
add_executable(ncnnTest squeezenet.cpp)
# 链接ncnn静态链接库
target_link_libraries(ncnnTest ncnn ${OpenCV_LIBS} /home/user/ncnn/build/install/lib/libncnn.a)
3 Windows 10下ncnn安装和使用
Win10下安装和ncnn最好都基于visual studio2015以上平台,本文用的vs2017。
3.1 Windows 10下ncnn编译安装
(1)protobuf编译
Win10下需要首先编译ncnn所使用到的protobuf库。Protobuf3.4.0下载路径为:??https://github.com/google/protobuf/archive/v3.4.0.zip??
然后选择vs2017自带的命令提示符工具,对于命令提示符工具选择,用于后期可能会用到opencv或者其他软件包。需要确定是x86还是x64平台,本文由于用的x64平台,选择适用于vs2017的x64本机工具。如下所示:
特别要注意的在编译文件前,一定要确定命令工具所使用的是x64还是x86。
protobuf编译步骤为:
cd <protobuf-root-dir>
mkdir build-vs2017
cd build-vs2017
cmake -G"NMake Makefiles" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%cd%/install -Dprotobuf_BUILD_TESTS=OFF -Dprotobuf_MSVC_STATIC_RUNTIME=OFF ../cmake
nmake
nmake install
上面protobuf-root-dir为你的protobuf文件路径,比如我的是D:/packages/protobuf-3.4.0,文件路径最好纯英文,protobuf设置的release格式。
(2)ncnn编译
下载ncnn源码,??https://github.com/Tencent/ncnn?? 然后用上面提到的开发工具进入ncnn安装即可,具体步骤如下:
cd <ncnn-root-dir>
mkdir build-vs2017
cd build-vs2017
cmake -G"NMake Makefiles" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%cd%/install -DProtobuf_INCLUDE_DIR=<protobuf-root-dir>/build-vs2017/install/include -DProtobuf_LIBRARIES=<protobuf-root-dir>/build-vs2017/install/lib/libprotobuf.lib -DProtobuf_PROTOC_EXECUTABLE=<protobuf-root-dir>/build-vs2017/install/bin/protoc.exe -DNCNN_VULKAN=OFF ..
nmake
nmake install
注意<protobuf-root-dir>要替换为你的protobuf文件绝对路径,比如我的是D:/packages/protobuf-3.4.0。DCMAKE_BUILD_TYPE=Release确定编译的是release文件。
3.2 Windows 10下ncnn使用
ncnn使用类似opencv在windows下的使用。打开vs2017建立工程文件。项目-属性-VC++目录。设置配置文件,VC++目录在包含目录下输入以下路径,调用opencv,ncnn,protobuf头文件。
d:\opencv\build\include\opencv
d:\opencv\build\include
d:\opencv\build\include\opencv2
d:\packages\ncnn\build-vs2017\install\include
d:\packages\ncnn\build-vs2017\install\include\ncnn
d:\packages\protobuf-3.4.0\build-vs2017\install\include
d:\packages\protobuf-3.4.0\build-vs2017\install\include\google
具体如图所示:
接着配置库目录,输入以下路径配置链接文件
d:\opencv\build\x64\vc15\lib
d:\packages\ncnn\build-vs2017\install\lib
d:\packages\protobuf-3.4.0\build-vs2017\install\lib
但是额外要配置配置Windows运行库目录,主要是要protobuf配置文件:
d:\packages\protobuf-3.4.0\build-vs2017\install\bin
最后链接器-输入-附加依赖项配置附加依赖项。Protobuf相关依赖项已经调用了其动态库,就不需要再调用了。
opencv_world341.lib
ncnn.lib
这样ncnn就配置好了。输入代码,直接ncnn/examples中选用一个示例调用cpp就行了。但是只能在release x64下调用ncnn,具体编译ncnn时所选择的编译模式。比如squeezenet.cpp,直接复制过来,设置param和bin文件路径就行了。其他模型依葫芦画瓢。
// Tencent is pleased to support the open source community by making ncnn available.
//
// Copyright (C) 2017 THL A29 Limited, a Tencent company. All rights reserved.
//
// Licensed under the BSD 3-Clause License (the "License"); you may not use this file except
// in compliance with the License. You may obtain a copy of the License at
//
// https://opensource.org/licenses/BSD-3-Clause
//
// Unless required by applicable law or agreed to in writing, software distributed
// under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
// specific language governing permissions and limitations under the License.
#include <stdio.h>
#include <algorithm>
#include <vector>
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
// linux平台调用
// #include "platform.h"
// #include "net.h"
// windows平台调用
#include <net.h>
#include <platform.h>
#if NCNN_VULKAN
#include "gpu.h"
#endif // NCNN_VULKAN
static int detect_squeezenet(const cv::Mat& bgr, std::vector<float>& cls_scores)
{
ncnn::Net squeezenet;
#if NCNN_VULKAN
squeezenet.opt.use_vulkan_compute = true;
#endif // NCNN_VULKAN
// squeezenet的ncnn模型文件路径
squeezenet.load_param("squeezenet_v1.1.param");
squeezenet.load_model("squeezenet_v1.1.bin");
ncnn::Mat in = ncnn::Mat::from_pixels_resize(bgr.data, ncnn::Mat::PIXEL_BGR, bgr.cols, bgr.rows, 227, 227);
const float mean_vals[3] = {104.f, 117.f, 123.f};
in.substract_mean_normalize(mean_vals, 0);
ncnn::Extractor ex = squeezenet.create_extractor();
ex.input("data", in);
ncnn::Mat out;
ex.extract("prob", out);
cls_scores.resize(out.w);
for (int j=0; j<out.w; j++)
{
cls_scores[j] = out[j];
}
return 0;
}
static int print_topk(const std::vector<float>& cls_scores, int topk)
{
// partial sort topk with index
int size = cls_scores.size();
std::vector< std::pair<float, int> > vec;
vec.resize(size);
for (int i=0; i<size; i++)
{
vec[i] = std::make_pair(cls_scores[i], i);
}
std::partial_sort(vec.begin(), vec.begin() + topk, vec.end(),
std::greater< std::pair<float, int> >());
// print topk and score
for (int i=0; i<topk; i++)
{
float score = vec[i].first;
int index = vec[i].second;
fprintf(stderr, "%d = %f\n", index, score);
}
return 0;
}
int main()
{
String imagepath = "./image/cat.jpg";
cv::Mat m = cv::imread(imagepath);
if (m.empty())
{
fprintf(stderr, "cv::imread %s failed\n", imagepath);
return -1;
}
#if NCNN_VULKAN
ncnn::create_gpu_instance();
#endif // NCNN_VULKAN
std::vector<float> cls_scores;
detect_squeezenet(m, cls_scores);
#if NCNN_VULKAN
ncnn::destroy_gpu_instance();
#endif // NCNN_VULKAN
print_topk(cls_scores, 3);
return 0;
}
4 参考
- ??https://github.com/Tencent/ncnn/wiki/how-to-build#build-for-linux-x86??
- ??https://github.com/Tencent/ncnn/wiki/ncnn??
相关推荐
- 4万多吨豪华游轮遇险 竟是因为这个原因……
-
(观察者网讯)4.7万吨豪华游轮搁浅,竟是因为油量太低?据观察者网此前报道,挪威游轮“维京天空”号上周六(23日)在挪威近海发生引擎故障搁浅。船上载有1300多人,其中28人受伤住院。经过数天的调...
- “菜鸟黑客”必用兵器之“渗透测试篇二”
-
"菜鸟黑客"必用兵器之"渗透测试篇二"上篇文章主要针对伙伴们对"渗透测试"应该如何学习?"渗透测试"的基本流程?本篇文章继续上次的分享,接着介绍一下黑客们常用的渗透测试工具有哪些?以及用实验环境让大家...
- 科幻春晚丨《震动羽翼说“Hello”》两万年星间飞行,探测器对地球的最终告白
-
作者|藤井太洋译者|祝力新【编者按】2021年科幻春晚的最后一篇小说,来自大家喜爱的日本科幻作家藤井太洋。小说将视角放在一颗太空探测器上,延续了他一贯的浪漫风格。...
- 麦子陪你做作业(二):KEGG通路数据库的正确打开姿势
-
作者:麦子KEGG是通路数据库中最庞大的,涵盖基因组网络信息,主要注释基因的功能和调控关系。当我们选到了合适的候选分子,单变量研究也已做完,接着研究机制的时便可使用到它。你需要了解你的分子目前已有哪些...
- 知存科技王绍迪:突破存储墙瓶颈,详解存算一体架构优势
-
智东西(公众号:zhidxcom)编辑|韦世玮智东西6月5日消息,近日,在落幕不久的GTIC2021嵌入式AI创新峰会上,知存科技CEO王绍迪博士以《存算一体AI芯片:AIoT设备的算力新选择》...
- 每日新闻播报(September 14)_每日新闻播报英文
-
AnOscarstatuestandscoveredwithplasticduringpreparationsleadinguptothe87thAcademyAward...
- 香港新巴城巴开放实时到站数据 供科技界研发使用
-
中新网3月22日电据香港《明报》报道,香港特区政府致力推动智慧城市,鼓励公私营机构开放数据,以便科技界研发使用。香港运输署21日与新巴及城巴(两巴)公司签署谅解备忘录,两巴将于2019年第3季度,开...
- 5款不容错过的APP: Red Bull Alert,Flipagram,WifiMapper
-
本周有不少非常出色的app推出,鸵鸟电台做了一个小合集。亮相本周榜单的有WifiMapper's安卓版的app,其中包含了RedBull的一款新型闹钟,还有一款可爱的怪物主题益智游戏。一起来看看我...
- Qt动画效果展示_qt显示图片
-
今天在这篇博文中,主要实践Qt动画,做一个实例来讲解Qt动画使用,其界面如下图所示(由于没有录制为gif动画图片,所以请各位下载查看效果):该程序使用应用程序单窗口,主窗口继承于QMainWindow...
- 如何从0到1设计实现一门自己的脚本语言
-
作者:dong...
- 三年级语文上册 仿写句子 需要的直接下载打印吧
-
描写秋天的好句好段1.秋天来了,山野变成了美丽的图画。苹果露出红红的脸庞,梨树挂起金黄的灯笼,高粱举起了燃烧的火把。大雁在天空一会儿写“人”字,一会儿写“一”字。2.花园里,菊花争奇斗艳,红的似火,粉...
- C++|那些一看就很简洁、优雅、经典的小代码段
-
目录0等概率随机洗牌:1大小写转换2字符串复制...
- 二年级上册语文必考句子仿写,家长打印,孩子照着练
-
二年级上册语文必考句子仿写,家长打印,孩子照着练。具体如下:...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)