Ubuntu16.04---腾讯NCNN框架入门到应用

前端之家收集整理的这篇文章主要介绍了Ubuntu16.04---腾讯NCNN框架入门到应用前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

Ubuntu16.04—腾讯NCNN框架入门到应用


前言

两天前腾讯发布NCNN深度学习框架后,发现可能有些同学对如何使用这些框架并不是十分的了解,一方面这是一个新的框架,另一方面Tencent出的文档对一些细节没有叙述,可能大牛们觉得很容易的步骤,我们往往会卡在那里,所以写下这篇博客来帮助一些对NCNN框架不是很熟悉的人来快速入门。
NCNN源码的地址为https://github.com/Tencent/ncnn


在Ubuntu上安装NCNN

1. 下载编译源码

  1. ruyiwei@ruyiwei:~/code$ git clone https://github.com/Tencent/ncnn

下载完成后,需要对源码进行编译

  1. cd ncnn
  2. mkdir build && cd build
  3. cmake ..
  4. make -j
  5. make install

执行完毕后我们可以看到

  1. Install the project...
  2. -- Install configuration: "release" -- Installing: /home/ruyiwei/code/ncnn/build/install/lib/libncnn.a -- Installing: /home/ruyiwei/code/ncnn/build/install/include/blob.h -- Installing: /home/ruyiwei/code/ncnn/build/install/include/cpu.h -- Installing: /home/ruyiwei/code/ncnn/build/install/include/layer.h -- Installing: /home/ruyiwei/code/ncnn/build/install/include/mat.h -- Installing: /home/ruyiwei/code/ncnn/build/install/include/net.h -- Installing: /home/ruyiwei/code/ncnn/build/install/include/opencv.h -- Installing: /home/ruyiwei/code/ncnn/build/install/include/platform.h

我们进入 ncnn/build/tools 目录下,如下所示,我们可以看到已经生成了 caffe2ncnn 可ncnn2mem这两个可执行文件,这两个可执行文件的作用是将caffe模型生成ncnn 模型,并且对模型进行加密,具体的使用方法我门在下一节讲述。

  1. ruyiwei@ruyiwei:~/code/ncnn/build/tools$ ll
  2. total 3024
  3. drwxrwxr-x 3 ruyiwei ruyiwei 4096 7 27 15:36 ./
  4. drwxrwxr-x 6 ruyiwei ruyiwei 4096 7 27 15:36 ../
  5. -rwxrwxr-x 1 ruyiwei ruyiwei 833720 7 27 15:36 caffe2ncnn*
  6. -rw-rw-r-- 1 ruyiwei ruyiwei 1102486 7 27 15:36 caffe.pb.cc
  7. -rw-rw-r-- 1 ruyiwei ruyiwei 894690 7 27 15:36 caffe.pb.h
  8. drwxrwxr-x 4 ruyiwei ruyiwei 4096 7 27 15:36 CMakeFiles/
  9. -rw-rw-r-- 1 ruyiwei ruyiwei 1018 7 27 15:36 cmake_install.cmake
  10. -rw-rw-r-- 1 ruyiwei ruyiwei 9353 7 27 15:36 Makefile
  11. -rwxrwxr-x 1 ruyiwei ruyiwei 228032 7 27 15:36 ncnn2mem*

2. 将caffe下AlexNet网络模型转换为NCNN模型

我们在测试的过程中需要caffemodel以及deploy.prototxt,所以我们再将caffe模型转换为NCNN模型的时候,同样也需要caffemodel以及deploy.prototxt这两个文件,为了方便,我们使用AlexNet为例讲解。

AlexNet 的 deploy.prototxt 可以在这里下载 https://github.com/BVLC/caffe/tree/master/models/bvlc_alexnet
AlexNet 的 caffemodel 可以在这里下载 http://dl.caffe.berkeleyvision.org/bvlc_alexnet.caffemodel

由于NCNN提供的转换工具只支持转换新版的caffe模型,所以我们需要利用caffe自带的工具将旧版的caffe模型转换为新版的caffe模型后,在将新版本的模型转换为NCNN模型.

  1. 旧版本caffe模型->新版本caffe模型->NCNN模型

2.1 旧版caffe模型转新版caffe模型

我们执行如下命令.[要记得修改路径]

  1. ruyiwei@ruyiwei:~/code/ncnn/build/tools$ ~/caffe/build/tools/upgrade_net_proto_text deploy.prototxt new_deplpy.prototxt
  1. ruyiwei@ruyiwei:~/code/ncnn/build/tools$ ~/caffe/build/tools/upgrade_net_proto_binary bvlc_AlexNet.caffemodel new_bvlc_AlexNet.caffemodel

上面的命令需要根据自己的caffe位置进行修改

执行后,就可以生成新的caffe模型.

因为我们每次检测一张图片,所以要对新生成的deploy.prototxt进行修改:第一个 dim 设为 1

  1. layer {
  2. name: "data"
  3. type: "Input"
  4. top: "data"
  5. input_param { shape: { dim: 1 dim: 3 dim: 227 dim: 227 } }
  6. }

2.2 新版caffe模型转ncnn模型

  1. ruyiwei@ruyiwei:~/code/ncnn/build/tools$./caffe2ncnn new_deplpy.prototxt new_bvlc_AlexNet.caffemodel AlexNet.param AlexNet.bin

执行上面命令后就可以生成NCNN模型需要的param 与bin 文件.

  1. ruyiwei@ruyiwei:~/code/ncnn/build/tools$ ll
  2. total 717492
  3. drwxrwxr-x 3 ruyiwei ruyiwei 4096 7 27 16:13 ./
  4. drwxrwxr-x 6 ruyiwei ruyiwei 4096 7 27 15:36 ../
  5. -rw-rw-r-- 1 ruyiwei ruyiwei 243860928 7 27 16:13 AlexNet.bin
  6. -rw-rw-r-- 1 ruyiwei ruyiwei 1583 7 27 16:13 AlexNet.param
  7. -rw-rw-r-- 1 ruyiwei ruyiwei 243862414 7 27 09:28 bvlc_AlexNet.caffemodel
  8. -rwxrwxr-x 1 ruyiwei ruyiwei 833720 7 27 15:36 caffe2ncnn*
  9. -rw-rw-r-- 1 ruyiwei ruyiwei 1102486 7 27 15:36 caffe.pb.cc
  10. -rw-rw-r-- 1 ruyiwei ruyiwei 894690 7 27 15:36 caffe.pb.h
  11. drwxrwxr-x 4 ruyiwei ruyiwei 4096 7 27 15:36 CMakeFiles/
  12. -rw-rw-r-- 1 ruyiwei ruyiwei 1018 7 27 15:36 cmake_install.cmake
  13. -rw-rw-r-- 1 ruyiwei ruyiwei 3629 6 6 21:40 deploy.prototxt
  14. -rw-rw-r-- 1 ruyiwei ruyiwei 9353 7 27 15:36 Makefile
  15. -rwxrwxr-x 1 ruyiwei ruyiwei 228032 7 27 15:36 ncnn2mem*
  16. -rw-rw-r-- 1 ruyiwei ruyiwei 243862660 7 27 16:03 new_bvlc_AlexNet.caffemodel
  17. -rw-r--r-- 1 ruyiwei ruyiwei 3662 7 27 16:03 new_deplpy.prototxt

3. 对模型参数加密

得到的AlexNet.param是明文可见的,往往发布的过程需要对这些文件进行加密,NCNN提供了对应的加密工具,ncnn2mem,

  1. ruyiwei@ruyiwei:~/code/ncnn/build/tools$ ./ncnn2mem AlexNet.param AlexNet.bin AlexNet.id.h AlexNet.mem.h

最后可以生成 AlexNet.param.bin 这样的二进制加密文件.

对于加密文件的读取也和原来不同,在源码中,非加密param读取方式为

  1. ncnn::Net net;
  2. net.load_param("AlexNet.param");
  3. net.load_model("AlexNet.bin");

加密param.bin读取方式为

  1. ncnn::Net net;
  2. net.load_param_bin("AlexNet.param.bin");
  3. net.load_model("AlexNet.bin");

4. 编译NCNN例程

前面介绍了如何将caffe模型转为NCNN模型并且加密,最后我们来编译NCNN的例程,这样可以更直观的运行或者理解NCNN.
首先我们需要进入ncnn/examples目录
新建一个makefile,内容如下,最重要的是,NCNN例程序支持opencv2,不支持opencv3.

  1. NCNN = /home/ruyiwei/code/ncnn
  2.  
  3. OPENCV = /home/ruyiwei/Downloads/opencv-2.4.13
  4.  
  5. INCPATH = -I${NCNN}/build/install/include \
  6. -I${OPENCV}/modules/objdetect/include \
  7. -I${OPENCV}/modules/highgui/include \
  8. -I${OPENCV}/modules/imgproc/include \
  9. -I${OPENCV}/modules/core/include
  10.  
  11. LIBS = -lopencv_core -lopencv_highgui -lopencv_imgproc \
  12. -fopenmp -pthread
  13.  
  14. LIBPATH = -L${OPENCV}/build/lib
  15.  
  16. %:%.cpp
  17. $(CXX) $(INCPATH) $(LIBPATH) $^ ${NCNN}/build/install/lib/libncnn.a $(LIBS) -o $@

在当前目录执行

  1. ruyiwei@ruyiwei:~/code/ncnn/examples$ ./squeezenet test.jpg

可得到如下结果

  1. ruyiwei@ruyiwei:~/code/ncnn/examples$ ./squeezenet test.jpg
  2. 404 = 0.990161
  3. 908 = 0.004498
  4. 405 = 0.004008

test.jpg 为下图所示:

为了可以更直观的显示,我们修改代码如下:

  1. #include <stdio.h>
  2. #include <algorithm>
  3. #include <vector>
  4. #include <opencv2/core/core.hpp>
  5. #include <opencv2/highgui/highgui.hpp>
  6. using namespace std;
  7. using namespace cv;
  8. #include "net.h"
  9.  
  10. static int detect_squeezenet(const cv::Mat& bgr,std::vector<float>& cls_scores)
  11. {
  12. ncnn::Net squeezenet;
  13. squeezenet.load_param("squeezenet_v1.1.param");
  14. squeezenet.load_model("squeezenet_v1.1.bin");
  15.  
  16. ncnn::Mat in = ncnn::Mat::from_pixels_resize(bgr.data,ncnn::Mat::PIXEL_BGR,bgr.cols,bgr.rows,227,227);
  17.  
  18. const float mean_vals[3] = {104.f,117.f,123.f};
  19. in.substract_mean_normalize(mean_vals,0);
  20.  
  21. ncnn::Extractor ex = squeezenet.create_extractor();
  22. ex.set_light_mode(true);
  23.  
  24. ex.input("data",in);
  25.  
  26. ncnn::Mat out;
  27. ex.extract("prob",out);
  28.  
  29. cls_scores.resize(out.c);
  30. for (int j=0; j<out.c; j++)
  31. {
  32. const float* prob = out.data + out.cstep * j;
  33. cls_scores[j] = prob[0];
  34. }
  35.  
  36. return 0;
  37. }
  38.  
  39. static int print_topk(const std::vector<float>& cls_scores,int topk,vector<int>& index_result,vector<float>& score_result)
  40. {
  41. // partial sort topk with index
  42. int size = cls_scores.size();
  43. std::vector< std::pair<float,int> > vec;
  44. vec.resize(size);
  45. for (int i=0; i<size; i++)
  46. {
  47. vec[i] = std::make_pair(cls_scores[i],i);
  48. }
  49.  
  50. std::partial_sort(vec.begin(),vec.begin() + topk,vec.end(),std::greater< std::pair<float,int> >());
  51.  
  52. // print topk and score
  53. for (int i=0; i<topk; i++)
  54. {
  55. float score = vec[i].first;
  56. int index = vec[i].second;
  57. index_result.push_back(index);
  58. score_result.push_back(score);
  59.  
  60. //fprintf(stderr,"%d = %f\n",index,score);
  61. }
  62.  
  63. return 0;
  64. }
  65.  
  66. static int load_labels(string path,vector<string>& labels)
  67. {
  68. FILE* fp = fopen(path.c_str(),"r");
  69.  
  70. while (!feof(fp))
  71. {
  72. char str[1024];
  73. fgets(str,1024,fp); //¶ÁÈ¡Ò»ÐÐ
  74. string str_s(str);
  75.  
  76. if (str_s.length() > 0)
  77. {
  78. for (int i = 0; i < str_s.length(); i++)
  79. {
  80. if (str_s[i] == ' ')
  81. {
  82. string strr = str_s.substr(i,str_s.length() - i - 1);
  83. labels.push_back(strr);
  84. i = str_s.length();
  85. }
  86. }
  87. }
  88. }
  89. return 0;
  90. }
  91.  
  92.  
  93. int main(int argc,char** argv)
  94. {
  95. const char* imagepath = argv[1];
  96. vector<string> labels;
  97. load_labels("synset_words.txt",labels);
  98. cv::Mat m = cv::imread(imagepath,CV_LOAD_IMAGE_COLOR);
  99. if (m.empty())
  100. {
  101. fprintf(stderr,"cv::imread %s Failed\n",imagepath);
  102. return -1;
  103. }
  104.  
  105. std::vector<float> cls_scores;
  106. detect_squeezenet(m,cls_scores);
  107.  
  108. vector<int> index;
  109. vector<float> score;
  110. print_topk(cls_scores,3,score);
  111.  
  112.  
  113. for (int i = 0; i < index.size(); i++)
  114. {
  115. cv::putText(m,labels[index[i]],Point(50,50 + 30 * i),CV_FONT_HERSHEY_SIMPLEX,1.2,Scalar(0,100,200),2,8);
  116. }
  117.  
  118. imshow("m",m);
  119. imwrite("test_result.jpg",m);
  120. waitKey(0);
  121.  
  122. return 0;
  123. }

这样就可以直观的显示出具体的类别,而不是数字,结果如下:

感谢

https://github.com/guozhongluo/ncnn-vs2015-examples-demo
https://github.com/Tencent/ncnn/wiki

猜你在找的Ubuntu相关文章