注:本文同步发布于微信公众号:stringwu的互联网杂谈TNN iOS 非图像模型入门指南

1 背景

TNN是腾讯优图实验室开源的高性能、轻量级神经网络推理框架TNN,github上也有比较详细的例子来说明如何在端上运行图像类的模型,但demo 更多是图像类相关的示例,而且里面做了一层层的封装,很难让一个初学者直接上手一步步构建出可推理的结果,

本文主要从初学者的角度出发,按照TNNAPI文档一步步构建出非图像模型的入门文档。(本文不再详述如何编译和集成TNN工程,有需要的同学可直接参考Demo文档);

2 构建

TNN的推理流程主要包括模型的解析,网络构建,输入设定,输出获取

2.1 模型解析

模型文件包括两个部分:

  • *.tnnmodel
  • *.tnnproto

模型解析的步骤包括:

  • 获取模型文件的路径
  • 解析文件内容
  • 初始化模型
	// 获取模型文件 (提前把对应的模型文件集成到工程中)
auto model_path = [[NSBundle mainBundle] pathForResource:@"model/recommend/recommend.tnnmodel"
ofType:nil];
auto proto_path = [[NSBundle mainBundle] pathForResource:@"model/recommend/recommend.tnnproto" ofType:nil]; //解析文件内容
string proto_content = [NSString stringWithContentsOfFile:proto_path encoding:NSUTF8StringEncoding error:nil].UTF8String; string model_content = [data_mode length] > 0 ? string((const char *)[data_mode bytes], [data_mode length]) : ""; //模型的配置
TNN_NS::ModelConfig model_config;
model_config.model_type = TNN_NS::MODEL_TYPE_TNN; // 指定模型的类型为TNN
model_config.params.push_back(proto_content);
model_config.params.push_back(model_content); auto tnn = std::make_shared<TNN_NS::TNN>(); //实例化TNN 的实例
//初始化模型
Status ret = tnn->Init(model_config);
//结果为TNN_OK时才为模型初始化成功
if (ret != TNN_OK) {
return;
}

2.2 网络构建

网络的构建需要配置TNN_NS::NetworkConfig,这个配置需要指定device_typelibrary_path,在iOS中的device_type正常是使用TNN_NS::DEVICE_ARMTNN_NS::DEVICE_METAL就可以了,但笔者在实际尝试时,发现device_type指定这两个类型都没有办法正常跑通,后与TNN相关同学咨询请教后,使用了TNN_NS::DEVICE_NAIVE才正常跑通,具体的原因TNN的同学还在帮忙定位中。(如果发现数据正常时,流程没有办法跑通的话,可以多换几个device_type看看)

  auto library_path = [[NSBundle mainBundle] pathForResource:@"tnn.metallib" ofType:nil];
//shape数据
auto shape_path = [[NSBundle mainBundle] pathForResource:@"model/recommend/tnn_input.json" ofType:nil];
//将shape数据转成NSDictionary
NSData *data_shape = [NSData dataWithContentsOfFile:shape_path];
NSDictionary * dictionary_shape = [NSJSONSerialization JSONObjectWithData:data_shape options:NSJSONReadingMutableLeaves error:nil];
//构造出TNN可识别的shape数据
InputShapesMap sdkInputShape = {};
for (NSString *key in dictionary_shape) {
NSDictionary*tmpValue = dictionary_shape[key];
NSLog(@"InputShapesMap: println key %@****",key);
NSInteger int1 = [tmpValue[@"dim1"]intValue];
NSInteger int2 = [tmpValue[@"dim2"]intValue];
NSLog(@"InputShapesMap222: println key %ld,%ld****",int1,int2);
TNN_NS::DimsVector nc = {(int)int1,(int)int2};
sdkInputShape.insert(std::pair<std::string, TNN_NS::DimsVector>(std::string(key.UTF8String),nc));
} // 构造出net_config
TNN_NS::NetworkConfig net_config;
net_config.device_type = TNN_NS::DEVICE_NAIVE; // 指定device_type,如果跑不成功,可以多换几个type试试
net_config.library_path = {library_path.UTF8String}; TNN_NS::Status error;
//构造出TNN网络对象
auto net_instace = tnn->CreateInst(net_config, error,sdkOptions->input_shapes);
//结果为TNN_OK时才为网络构建成功
if (error != TNN_OK) {
return;
}

2.3 输入设定

输入设定主要是通过TNN的方法 SetInputMat 来完成的;

//从文件获取模型的输入
auto mock_input_path = [[NSBundle mainBundle] pathForResource:@"model/recommend/mock_input.json" ofType:nil];
//将数据转换成NSDictionary
NSData *data_mock = [NSData dataWithContentsOfFile:mock_input_path];
NSDictionary * dictionary_mock = [NSJSONSerialization JSONObjectWithData:data_mock options:NSJSONReadingMutableLeaves error:nil]; //将数据转换成TNN的输入格式
std:map<std::string,std::vector<float>> inputDatas = {};
for (NSString *key in dictionary_mock) {
NSArray *valueArray = dictionary_mock[key];
__block std::vector<float> valueVector ={};
valueVector.reserve([valueArray count]);
[valueArray enumerateObjectsUsingBlock:^( id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
valueVector.push_back([obj floatValue]);
}];
inputDatas[[key UTF8String]] = valueVector;
} BlobMap blob_map;
net_instace ->GetAllInputBlobs(blob_map);
//遍历所有的输入key
for (const auto&item : blob_map) {
//获取对应的key
std::string name = item.first;
//获取key对应的输入数据
std::vector<float> tmpItem = inputDatas.at(name);
//获取key对应的shape数据
DimsVector shape = sdkOptions->input_shapes[name];
//构造出input_mat
auto input_mat = std::make_shared<TNN_NS::Mat>(net_config.device_type,TNN_NS::NCHW_FLOAT,shape,tmpItem.data()); MatConvertParam input_convert_params = TNN_NS::MatConvertParam();
//把输入数据通过SetInputMat方法给到TNN引擎
auto status = net_instace->SetInputMat(input_mat,input_convert_params,name);
//结果为TNN_OK时才为设置输入成功
if (status != TNN_OK) {
NSLog(@"setInputmat error ");
return;
} } //执行推理
auto forwardStatus = net_instace->Forward();
if (forwardStatus != TNN_OK) {
NSLog(@"forwardStatus error ");
return;
}

2.4 输出获取

获取推理的结果,是通过TNNGetAllOutputBlobs接口来完成


BlobMap out_blob_map;
//获取所有的输出key
net_instace ->GetAllOutputBlobs(out_blob_map);
//所有的结果都输出放在 mat_map里面
std::map<std::string, std::shared_ptr<TNN_NS::Mat> > mat_map = {};
//遍历所有输出key
for (const auto&item: out_blob_map) {
auto name = item.first;
MatConvertParam output_convert_params = TNN_NS::MatConvertParam();
std::shared_ptr<TNN_NS::Mat> output_mat = nullptr;
auto status = net_instace->GetOutputMat(output_mat,output_convert_params,name,net_config.device_type);
if (status != TNN_OK) {
NSLog(@"outPutError error ");
return;
}
mat_map[name] = output_mat;
NSLog(@"outPutname %@",[NSString stringWithFormat:@"%s", name.c_str()]);
}

在得到输出结果的map后mat_map后,就可以根据业务的情况进行结果的解析,本文使用的模型最终的结果是一个float值

//推荐模型的推理结果是存在在pred字段里面;
auto scores = mat_map["pred"];
auto dims = scores ->GetDims();
if (dims.size() <= 0) {
NSLog(@"scores dims error ");
return;
} //获取当前的推理结果
float *score_data = static_cast<float*>(scores->GetData());
NSLog(@"scores is %f ",score_data[0]);

TNN iOS非图像模型入门的更多相关文章

  1. iOS 非ARC基本内存管理系列 -手把手教你ARC——iOS/Mac开发ARC入门和使用(转)

    手把手教你ARC——iOS/Mac开发ARC入门和使用 Revolution of Objective-c 本文部分实例取自iOS 5 Toturail一书中关于ARC的教程和公开内容,仅用于技术交流 ...

  2. 转 iOS Core Animation 动画 入门学习(一)基础

    iOS Core Animation 动画 入门学习(一)基础 reference:https://developer.apple.com/library/ios/documentation/Coco ...

  3. GAN实战笔记——第二章自编码器生成模型入门

    自编码器生成模型入门 之所以讲解本章内容,原因有三. 生成模型对大多数人来说是一个全新的领域.大多数人一开始接触到的往往都是机器学习中的分类任务--也许因为它们更为直观:而生成模型试图生成看起来很逼真 ...

  4. iOS开发-UI 从入门到精通(三)

    iOS开发-UI 从入门到精通(三)是对 iOS开发-UI 从入门到精通(一)知识点的综合练习,搭建一个简单地登陆界面,增强实战经验,为以后做开发打下坚实的基础! ※在这里我们还要强调一下,开发环境和 ...

  5. iOS开发-UI 从入门到精通(二)

    iOS开发-UI 从入门到精通(二)是对 iOS开发-UI 从入门到精通(一)知识点的巩固,主要以习题练习为主,增强实战经验,为以后做开发打下坚实的基础! ※开发环境和注意事项: 1.前期iOS-UI ...

  6. iOS实现图像的反色,怀旧,色彩直方图效果

    反色是与原色叠加可以变为白色的颜色,即用白色(RGB:1.0,1.0,1.0)减去原色的颜色.比如说红色(RGB:1.0,0,0)的反色是青色(0,1.0,1.0).在OPENGL ES中为1. 通过 ...

  7. Python的异步编程[0] -> 协程[1] -> 使用协程建立自己的异步非阻塞模型

    使用协程建立自己的异步非阻塞模型 接下来例子中,将使用纯粹的Python编码搭建一个异步模型,相当于自己构建的一个asyncio模块,这也许能对asyncio模块底层实现的理解有更大的帮助.主要参考为 ...

  8. IO阻塞模型、IO非阻塞模型、多路复用IO模型

    IO操作主要包括两类: 本地IO 网络IO 本地IO:本地IO是指本地的文件读取等操作,本地IO的优化主要是在操作系统中进行,我们对于本地IO的优化作用十分有限 网络IO:网络IO指的是在进行网络操作 ...

  9. JVM内存模型入门

    JVM内存模型入门 本文是学习笔记,原文地址在:https://www.bilibili.com/video/av62009886 综述 其实没有太多新东西 JVM主要分为五个区域:栈区.堆区.本地方 ...

随机推荐

  1. HTML5-CSS(四)

    一.CSS3 渐变效果 (1)CSS3 提供了 linear-gradient 属性实现背景颜色的渐变功能.首先,我们先看一下它的样式表,如下: //两个必须参数background-image: l ...

  2. 如何使用Scala的ClassTag

    Scala官方文档中对于ClassTag的定义如下: ClassTag[T]保存着在运行时被JVM擦除的类型T的信息.当我们在运行时想获得被实例化的Array的类型信息的时候,这个特性会比较有用. 下 ...

  3. 文件上传靶机DVWA和upload-labs

    DVWA靶机 LOW <?php phpinfo() ?> 上传文件 Medium级别 修改Content-Type: application/octet-stream的值为jpg的格式为 ...

  4. JBoss 5.x/6.x 反序列化漏洞(CVE-2017-12149)

    检测

  5. 浅谈MySQL与mongodb的区别

    讨论MySQL与mongodb使用上的区别以及可能适用的应用场景,不深入到数据库的实现细节方面.鉴于个人水平有限,文章可能存在错误之处,希望各位指正. 代码编写 mongodb支持reactor,可以 ...

  6. 痞子衡嵌入式:ARM Cortex-M内核那些事(9.1)- 存储保护(MPU - PMSAv6/7)

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是ARM Cortex-M存储保护模块(MPU). <ARM Cortex-M内核MCU开发那些事>的内核篇连载最早是 201 ...

  7. noi linux 2.0 体验

    一.起因 下午,我打开 noi 官网准备报名 csp j/s,一看官网展板:"noi linux 2.0 发布" 我就兴奋了起来.(9 月 1 日起开始使用, 也就意味着 csp ...

  8. python grequest模块使用备忘录

    手里上有一批链接,需要检查他们是否已经被删除.本来是想用多线程的,但是考虑了下一个是实现起来稍繁琐.而且性能不理想,单机基本超过10线程基本上就没有太多增益了. 所以考虑了下,还是决定用异步IO. 在 ...

  9. Burp - Turbo Intruder

    Turbo Intruder 基础使用总结,把Python代码都记录下,要是有啥骚姿势,求各位师傅交流. 个人感觉超强的一款Burp插件,反正超快 Link: https://github.com/P ...

  10. SpringBoot 如何进行参数校验,老鸟们都这么玩的!

    大家好,我是飘渺. 前几天写了一篇 SpringBoot如何统一后端返回格式?老鸟们都是这样玩的! 阅读效果还不错,而且被很多号主都转载过,今天我们继续第二篇,来聊聊在SprinBoot中如何集成参数 ...