飞桨上线万能转换小工具,教你玩转TensorFlow、Caffe等模型迁移

本文作者:飞桨开发者说成员Charlotte 0Hr每天发布大量与生活相关的资讯平台

百度推出飞桨(PaddlePaddle)后,不少开发者开始转向国内的深度学习框架。但是从代码的转移谈何容易,之前的工作重写一遍不太现实,成千上万行代码的手工转换等于是在做一次二次开发。0Hr每天发布大量与生活相关的资讯平台

现在,有个好消息:无论Caffe、TensorFlow、ONNX都可以轻松迁移到飞桨平台上。虽然目前还不直接迁移PyTorch模型,但PyTorch本身支持导出为ONNX模型,等于间接对该平台提供了支持。0Hr每天发布大量与生活相关的资讯平台

然而,有人还对存在疑惑:不同框架之间的API有没有差异?整个迁移过程如何操作,步骤复杂吗?迁移后如何保证精度的损失在可接受的范围内?0Hr每天发布大量与生活相关的资讯平台

大家会考虑很多问题,而问题再多,归纳一下,无外乎以下几点:0Hr每天发布大量与生活相关的资讯平台

  1. API差异:模型的实现方式如何迁移,不同框架之间的API有没有差异?如何避免这些差异带来的模型效果的差异?
  2. 预测方式差异:转换后的模型如何预测?预测的效果与转换前的模型差异如何?

X2Paddle(Github见参考1),可以支持主流深度学习框架模型转换至飞桨,包括Caffe、Tensorflow、onnx等模型直接转换为Paddle Fluid可加载的预测模型,并且还提供了这三大主流框架间的API差异比较,方便我们在自己直接复现模型时对比API之间的差异,深入理解API的实现方式从而降低模型迁移带来的损失。0Hr每天发布大量与生活相关的资讯平台

下面以TensorFlow转换成Paddle Fluid模型为例,详细讲讲如何实现模型的迁移。0Hr每天发布大量与生活相关的资讯平台

TensorFlow-Fluid 的API差异 0Hr每天发布大量与生活相关的资讯平台

在深度学习入门过程中,大家常见的就是手写数字识别这个demo,下面是一份最简单的实现手写数字识别的代码:0Hr每天发布大量与生活相关的资讯平台

fromtensorflow.examples.tutorials.mnist importinput_data 0Hr每天发布大量与生活相关的资讯平台

importtensorflow astf 0Hr每天发布大量与生活相关的资讯平台

mnist = input_data.read_data_sets( "MNIST_data/", one_hot= True) 0Hr每天发布大量与生活相关的资讯平台

x = tf.placeholder(tf.float32, [ None, 784]) 0Hr每天发布大量与生活相关的资讯平台

W = tf.Variable(tf.zeros([ 784, 10])) 0Hr每天发布大量与生活相关的资讯平台

b = tf.Variable(tf.zeros([ 10])) 0Hr每天发布大量与生活相关的资讯平台

y = tf.nn.softmax(tf.matmul(x, W) + b) 0Hr每天发布大量与生活相关的资讯平台

y_ = tf.placeholder( "float", [ None, 10]) 0Hr每天发布大量与生活相关的资讯平台

cross_entropy = tf.reduce_sum(tf.nn.softmax_cross_entropy_with_logits(logits = y,labels = y_)) 0Hr每天发布大量与生活相关的资讯平台

train_step = tf.train.GradientDescentOptimizer( 0.01).minimize(cross_entropy) 0Hr每天发布大量与生活相关的资讯平台

init = tf.global_variables_initializer 0Hr每天发布大量与生活相关的资讯平台

sess = tf.Session 0Hr每天发布大量与生活相关的资讯平台

sess.run(init) 0Hr每天发布大量与生活相关的资讯平台

fori inrange( 1, 1000): 0Hr每天发布大量与生活相关的资讯平台

batch_xs, batch_ys = mnist.train.next_batch( 100) 0Hr每天发布大量与生活相关的资讯平台

sess.run(train_step, feed_dict={x: batch_xs, y_: batch_ys}) 0Hr每天发布大量与生活相关的资讯平台

correct_prediction = tf.equal(tf.argmax(y, 1), tf.argmax(y_, 1)) 0Hr每天发布大量与生活相关的资讯平台

accuracy = tf.reduce_mean(tf.cast(correct_prediction, 'float')) 0Hr每天发布大量与生活相关的资讯平台

print(sess.run(accuracy, feed_dict={x: mnist.test.images, y_: mnist.test.labels})) 0Hr每天发布大量与生活相关的资讯平台

大家看这段代码里,第一步是导入mnist数据集,然后设置了一个占位符x来表示输入的图片数据,再设置两个变量w和b,分别表示权重和偏置来计算,最后通过softmax计算得到输出的y值,而我们真实的label则是变量y_ 。0Hr每天发布大量与生活相关的资讯平台

前向传播完成后,就可以计算预测值y与label y_之间的交叉熵。0Hr每天发布大量与生活相关的资讯平台

再选择合适的优化函数,此处为梯度下降,最后启动一个Session,把数据按batch灌进去,计算acc即可得到准确率。0Hr每天发布大量与生活相关的资讯平台

这是一段非常简单的代码,如果我们想把这段代码变成飞桨的代码,有人可能会认为非常麻烦,每一个实现的API还要一一去找对应的实现方式,但是这里,我可以告诉大家,不!用!这!么!麻!烦!因为在X2Paddle里有一份常用的Tensorflow对应Fluid的API表,(https://github.com/PaddlePaddle/X2Paddle/tree/master/tensorflow2fluid/doc),如下所示:0Hr每天发布大量与生活相关的资讯平台

0Hr每天发布大量与生活相关的资讯平台

对于常用的TensorFlow的API,都有相应的飞桨接口,如果两者的功能没有差异,则会标注功能一致,如果实现方式或者支持的功能、参数等有差异,即会标注“差异对比”,并详细注明。0Hr每天发布大量与生活相关的资讯平台

0Hr每天发布大量与生活相关的资讯平台

譬如,在上文这份非常简单的代码里,出现了这些TensorFlow的API:0Hr每天发布大量与生活相关的资讯平台

在出现的这些api里,大部分的功能都是一致的,只有两个功能不同,分别是tf.placeholder和tf.nn.softmax_cross_entropy_with_logits ,分别对应 fluid.layers.data 和 fluid.layers.softmax_with_cross_entropy . 我们来看看具体差异:0Hr每天发布大量与生活相关的资讯平台

tf.placeholder V.S fluid.layers.data 0Hr每天发布大量与生活相关的资讯平台

常用TensorFlow的同学对placeholder应该不陌生,中文翻译为占位符,什么意思呢?在TensorFlow 2.0以前,还是静态图的设计思想,整个设计理念是计算流图,在编写程序时,首先构筑整个系统的graph,代码并不会直接生效,这一点和python的其他数值计算库(如Numpy等)不同,graph为静态的,在实际的运行时,启动一个session,程序才会真正的运行。这样做的好处就是:避免反复地切换底层程序实际运行的上下文,tensorflow帮你优化整个系统的代码。我们知道,很多python程序的底层为C语言或者其他语言,执行一行脚本,就要切换一次,是有成本的,tensorflow通过计算流图的方式,可以帮你优化整个session需要执行的代码。0Hr每天发布大量与生活相关的资讯平台

在代码层面,每一个tensor值在graph上都是一个op,当我们将train数据分成一个个minibatch然后传入网络进行训练时,每一个minibatch都将是一个op,这样的话,一副graph上的op未免太多,也会产生巨大的开销;于是就有了tf.placeholder,我们每次可以将 一个minibatch传入到x = tf.placeholder(tf.float32,[None,32])上,下一次传入的x都替换掉上一次传入的x,这样就对于所有传入的minibatch x就只会产生一个op,不会产生其他多余的op,进而减少了graph的开销。0Hr每天发布大量与生活相关的资讯平台

参数对比 0Hr每天发布大量与生活相关的资讯平台

tf.placeholder0Hr每天发布大量与生活相关的资讯平台

tf.placeholder( 0Hr每天发布大量与生活相关的资讯平台

dtype, 0Hr每天发布大量与生活相关的资讯平台

shape= None, 0Hr每天发布大量与生活相关的资讯平台

name= None0Hr每天发布大量与生活相关的资讯平台

) 0Hr每天发布大量与生活相关的资讯平台

paddle.fluid.layers.data0Hr每天发布大量与生活相关的资讯平台

paddle.fluid.layers.data( 0Hr每天发布大量与生活相关的资讯平台

name, 0Hr每天发布大量与生活相关的资讯平台

shape, 0Hr每天发布大量与生活相关的资讯平台

append_batch_size= True, 0Hr每天发布大量与生活相关的资讯平台

dtype= 'float32', 0Hr每天发布大量与生活相关的资讯平台

lod_level= 0, 0Hr每天发布大量与生活相关的资讯平台

type=VarType.LOD_TENSOR, 0Hr每天发布大量与生活相关的资讯平台

stop_gradient= True) 0Hr每天发布大量与生活相关的资讯平台

从图中可以看到,飞桨的api参数更多,具体差异如下:0Hr每天发布大量与生活相关的资讯平台

  • Batch维度处理

TensorFlow: 对于shape中的batch维度,需要用户使用None指定;0Hr每天发布大量与生活相关的资讯平台

飞桨: 将第1维设置为-1表示batch维度;如若第1维为正数,则会默认在最前面插入batch维度,如若要避免batch维,可将参数append_batch_size设为False。0Hr每天发布大量与生活相关的资讯平台

  • 梯度是否回传

tensorflow和pytorch都支持对输入求梯度,在飞桨中直接设置stop_gradient = False即可。如果在某一层使用stop_gradient=True,那么这一层之前的层都会自动的stop_gradient=True,梯度不会参与回传,可以对某些不需要参与loss计算的信息设置为stop_gradient=True。对于含有BatchNormalization层的CNN网络,也可以对输入求梯度,如0Hr每天发布大量与生活相关的资讯平台

layers.data( 0Hr每天发布大量与生活相关的资讯平台

name= "data", 0Hr每天发布大量与生活相关的资讯平台

shape=[ 32, 3, 224, 224], 0Hr每天发布大量与生活相关的资讯平台

dtype= "int64", 0Hr每天发布大量与生活相关的资讯平台

append_batch_size= False, 0Hr每天发布大量与生活相关的资讯平台

stop_gradient= False) 0Hr每天发布大量与生活相关的资讯平台

tf.nn.softmax_cross_entropy_with_logits V.S fluid.layers.softmax_with_cross_entropy 参数对比 0Hr每天发布大量与生活相关的资讯平台

tf.nn.softmax_cross_entropy_with_logits( 0Hr每天发布大量与生活相关的资讯平台

_sentinel= None, 0Hr每天发布大量与生活相关的资讯平台

labels= None, 0Hr每天发布大量与生活相关的资讯平台

logits= None, 0Hr每天发布大量与生活相关的资讯平台

dim= -1, 0Hr每天发布大量与生活相关的资讯平台

name= None0Hr每天发布大量与生活相关的资讯平台

) 0Hr每天发布大量与生活相关的资讯平台

paddle.fluid.layers.softmax_with_cross_entropy0Hr每天发布大量与生活相关的资讯平台

paddle.fluid.layers.softmax_with_cross_entropy( 0Hr每天发布大量与生活相关的资讯平台

logits, 0Hr每天发布大量与生活相关的资讯平台

label, 0Hr每天发布大量与生活相关的资讯平台

soft_label= False, 0Hr每天发布大量与生活相关的资讯平台

ignore_index= -100, 0Hr每天发布大量与生活相关的资讯平台

numeric_stable_mode= False, 0Hr每天发布大量与生活相关的资讯平台

return_softmax= False0Hr每天发布大量与生活相关的资讯平台

) 0Hr每天发布大量与生活相关的资讯平台

功能差异0Hr每天发布大量与生活相关的资讯平台

标签类型0Hr每天发布大量与生活相关的资讯平台

TensorFlow:labels只能使用软标签,其shape为[batch, num_classes],表示样本在各个类别上的概率分布;0Hr每天发布大量与生活相关的资讯平台

飞桨:通过设置soft_label,可以选择软标签或者硬标签。当使用硬标签时,label的shape为[batch, 1],dtype为int64;当使用软标签时,其shape为[batch, num_classes],dtype为int64。0Hr每天发布大量与生活相关的资讯平台

返回值0Hr每天发布大量与生活相关的资讯平台

TensorFlow:返回batch中各个样本的log loss;0Hr每天发布大量与生活相关的资讯平台

飞桨:当return_softmax为False时,返回batch中各个样本的log loss;当return_softmax为True时,再额外返回logtis的归一化值。0Hr每天发布大量与生活相关的资讯平台

疑问点? 0Hr每天发布大量与生活相关的资讯平台

硬标签,即 one-hot label, 每个样本仅可分到一个类别0Hr每天发布大量与生活相关的资讯平台

numeric_stable_mode:这个参数是什么呢?标志位,指明是否使用一个具有更佳数学稳定性的算法。仅在 soft_label 为 False的GPU模式下生效. 若 soft_label 为 True 或者执行场所为CPU, 算法一直具有数学稳定性。注意使用稳定算法时速度可能会变慢。默认为 True。0Hr每天发布大量与生活相关的资讯平台

return_softmax: 指明是否额外返回一个softmax值, 同时返回交叉熵计算结果。默认为False。0Hr每天发布大量与生活相关的资讯平台

如果 return_softmax 为 False, 则返回交叉熵损失0Hr每天发布大量与生活相关的资讯平台

如果 return_softmax 为 True,则返回元组 (loss, softmax) ,其中交叉熵损失为形为[N x 1]的二维张量,softmax为[N x K]的二维张量0Hr每天发布大量与生活相关的资讯平台

代码示例 0Hr每天发布大量与生活相关的资讯平台

data = fluid.layers.data(name= 'data', shape=[ 128], dtype= 'float32') 0Hr每天发布大量与生活相关的资讯平台

label = fluid.layers.data(name= 'label', shape=[ 1], dtype= 'int64') 0Hr每天发布大量与生活相关的资讯平台

fc = fluid.layers.fc(input=data, size= 100) 0Hr每天发布大量与生活相关的资讯平台

out = fluid.layers.softmax_with_cross_entropy( 0Hr每天发布大量与生活相关的资讯平台

logits=fc, label=label) 0Hr每天发布大量与生活相关的资讯平台

所以通过API对应表,我们可以直接转换把TensorFlow代码转换成Paddle Fluid代码。但是如果现在项目已经上线了,代码几千行甚至上万行,或者已经训练出可预测的模型了,如果想要直接转换API是一件非常耗时耗精力的事情,有没有一种方法可以直接把训练好的可预测模型直接转换成另一种框架写的,只要转换后的损失精度在可接受的范围内,就可以直接替换。下面就讲讲训练好的模型如何迁移。0Hr每天发布大量与生活相关的资讯平台

模型迁移 0Hr每天发布大量与生活相关的资讯平台

VGG_16是CV领域的一个经典模型,我以tensorflow/models下的VGG_16为例,给大家展示如何将TensorFlow训练好的模型转换为飞桨模型。0Hr每天发布大量与生活相关的资讯平台

importurllib 0Hr每天发布大量与生活相关的资讯平台

importsys 0Hr每天发布大量与生活相关的资讯平台

defschedule(a, b, c):0Hr每天发布大量与生活相关的资讯平台

per = 100.0* a * b / c 0Hr每天发布大量与生活相关的资讯平台

per = int(per) 0Hr每天发布大量与生活相关的资讯平台

sys.stderr.write( "rDownload percentage %.2f%%"% per) 0Hr每天发布大量与生活相关的资讯平台

sys.stderr.flush 0Hr每天发布大量与生活相关的资讯平台

url = "http://download.tensorflow.org/models/vgg_16_2016_08_28.tar.gz"0Hr每天发布大量与生活相关的资讯平台

fetch = urllib.urlretrieve(url, "./vgg_16.tar.gz", schedule) 0Hr每天发布大量与生活相关的资讯平台

importtarfile 0Hr每天发布大量与生活相关的资讯平台

withtarfile.open( "./vgg_16.tar.gz", "r:gz") asf: 0Hr每天发布大量与生活相关的资讯平台

file_names = f.getnames 0Hr每天发布大量与生活相关的资讯平台

forfile_name infile_names: 0Hr每天发布大量与生活相关的资讯平台

f.extract(file_name, "./") 0Hr每天发布大量与生活相关的资讯平台

保存模型为checkpoint格式0Hr每天发布大量与生活相关的资讯平台

importtensorflow.contrib.slim asslim 0Hr每天发布大量与生活相关的资讯平台

fromtensorflow.contrib.slim.nets importvgg 0Hr每天发布大量与生活相关的资讯平台

importtensorflow astf 0Hr每天发布大量与生活相关的资讯平台

importnumpy 0Hr每天发布大量与生活相关的资讯平台

withtf.Session assess: 0Hr每天发布大量与生活相关的资讯平台

inputs = tf.placeholder(dtype=tf.float32, shape=[ None, 224, 224, 3], name= "inputs") 0Hr每天发布大量与生活相关的资讯平台

logits, endpoint = vgg.vgg_16(inputs, num_classes= 1000, is_training= False) 0Hr每天发布大量与生活相关的资讯平台

load_model = slim.assign_from_checkpoint_fn( "vgg_16.ckpt", slim.get_model_variables( "vgg_16")) 0Hr每天发布大量与生活相关的资讯平台

load_model(sess) 0Hr每天发布大量与生活相关的资讯平台

numpy.random.seed( 13) 0Hr每天发布大量与生活相关的资讯平台

data = numpy.random.rand( 5, 224, 224, 3) 0Hr每天发布大量与生活相关的资讯平台

input_tensor = sess.graph.get_tensor_by_name( "inputs:0") 0Hr每天发布大量与生活相关的资讯平台

output_tensor = sess.graph.get_tensor_by_name( "vgg_16/fc8/squeezed:0") 0Hr每天发布大量与生活相关的资讯平台

result = sess.run([output_tensor], {input_tensor:data}) 0Hr每天发布大量与生活相关的资讯平台

numpy.save( "tensorflow.npy", numpy.array(result)) 0Hr每天发布大量与生活相关的资讯平台

saver = tf.train.Saver 0Hr每天发布大量与生活相关的资讯平台

saver.save(sess, "./checkpoint/model") 0Hr每天发布大量与生活相关的资讯平台

TensorFlow2fluid目前支持checkpoint格式的模型或者是将网络结构和参数序列化的pb格式模型,上面下载的vgg_16.ckpt仅仅存储了模型参数,因此我们需要重新加载参数,并将网络结构和参数一起保存为checkpoint模型0Hr每天发布大量与生活相关的资讯平台

importtf2fluid.convert asconvert 0Hr每天发布大量与生活相关的资讯平台

importargparse 0Hr每天发布大量与生活相关的资讯平台

parser = convert._get_parser 0Hr每天发布大量与生活相关的资讯平台

parser.meta_file = "checkpoint/model.meta"0Hr每天发布大量与生活相关的资讯平台

parser.ckpt_dir = "checkpoint"0Hr每天发布大量与生活相关的资讯平台

parser.in_nodes = [ "inputs"] 0Hr每天发布大量与生活相关的资讯平台

parser.input_shape = [ "None,224,224,3"] 0Hr每天发布大量与生活相关的资讯平台

parser.output_nodes = [ "vgg_16/fc8/squeezed"] 0Hr每天发布大量与生活相关的资讯平台

parser.use_cuda = "True"0Hr每天发布大量与生活相关的资讯平台

parser.input_format = "NHWC"0Hr每天发布大量与生活相关的资讯平台

parser.save_dir = "paddle_model"0Hr每天发布大量与生活相关的资讯平台

convert.run(parser) 0Hr每天发布大量与生活相关的资讯平台

注意:部分OP在转换时,需要将参数写入文件;或者是运行tensorflow模型进行infer,获取tensor值。两种情况下均会消耗一定的时间用于IO或计算,对于后一种情况,0Hr每天发布大量与生活相关的资讯平台

打印输出log信息(截取部分)0Hr每天发布大量与生活相关的资讯平台

INFO:root:Loading tensorflow model... 0Hr每天发布大量与生活相关的资讯平台

INFO:tensorflow:Restoring parameters fromcheckpoint/model 0Hr每天发布大量与生活相关的资讯平台

INFO:tensorflow:Restoring parameters fromcheckpoint/model 0Hr每天发布大量与生活相关的资讯平台

INFO:root:Tensorflow model loaded! 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 1,CurrentNode:inputs 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 2,CurrentNode:vgg_16/conv1/conv1_1/weights 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 3,CurrentNode:vgg_16/conv1/conv1_1/biases 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 4,CurrentNode:vgg_16/conv1/conv1_2/weights 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 5,CurrentNode:vgg_16/conv1/conv1_2/biases 0Hr每天发布大量与生活相关的资讯平台

... 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 10,CurrentNode:vgg_16/conv3/conv3_1/weights 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 11,CurrentNode:vgg_16/conv3/conv3_1/biases 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 12,CurrentNode:vgg_16/conv3/conv3_2/weights 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 13,CurrentNode:vgg_16/conv3/conv3_2/biases 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 85,CurrentNode:vgg_16/fc8/BiasAdd 0Hr每天发布大量与生活相关的资讯平台

INFO:root:TotalNum: 86,TraslatedNum: 86,CurrentNode:vgg_16/fc8/squeezed 0Hr每天发布大量与生活相关的资讯平台

INFO:root:Model translated! 0Hr每天发布大量与生活相关的资讯平台

到这一步,我们已经把tensorflow/models下的vgg16模型转换成了Paddle Fluid 模型,转换后的模型与原模型的精度有损失吗?如何预测呢?来看下面。0Hr每天发布大量与生活相关的资讯平台

预测结果差异 0Hr每天发布大量与生活相关的资讯平台

上一步转换后的模型目录命名为“paddle_model”,在这里我们通过ml.ModelLoader把模型加载进来,注意转换后的飞桨模型的输出格式由NHWC转换为NCHW,所以我们需要对输入数据做一个转置。处理好数据后,即可通过model.inference来进行预测了。具体代码如下:0Hr每天发布大量与生活相关的资讯平台

importnumpy 0Hr每天发布大量与生活相关的资讯平台

importtf2fluid.model_loader asml 0Hr每天发布大量与生活相关的资讯平台

model = ml.ModelLoader( "paddle_model", use_cuda= False) 0Hr每天发布大量与生活相关的资讯平台

numpy.random.seed( 13) 0Hr每天发布大量与生活相关的资讯平台

data = numpy.random.rand( 5, 224, 224, 3).astype( "float32") 0Hr每天发布大量与生活相关的资讯平台

# NHWC -> NCHW0Hr每天发布大量与生活相关的资讯平台

data = numpy.transpose(data, ( 0, 3, 1, 2)) 0Hr每天发布大量与生活相关的资讯平台

results = model.inference(feed_dict={model.inputs[ 0]:data}) 0Hr每天发布大量与生活相关的资讯平台

numpy.save( "paddle.npy", numpy.array(results)) 0Hr每天发布大量与生活相关的资讯平台

对比模型损失 0Hr每天发布大量与生活相关的资讯平台

转换模型有一个问题始终避免不了,就是损失,从Tesorflow的模型转换为Paddle Fluid模型,如果模型的精度损失过大,那么转换模型实际上是没有意义的,只有损失的精度在我们可接受的范围内,模型转换才能被实际应用。在这里可以通过把两个模型文件加载进来后,通过numpy.fabs来求两个模型结果的差异。0Hr每天发布大量与生活相关的资讯平台

importnumpy 0Hr每天发布大量与生活相关的资讯平台

paddle_result = numpy.load( "paddle.npy") 0Hr每天发布大量与生活相关的资讯平台

tensorflow_result = numpy.load( "tensorflow.npy") 0Hr每天发布大量与生活相关的资讯平台

diff = numpy.fabs(paddle_result - tensorflow_result) 0Hr每天发布大量与生活相关的资讯平台

print(numpy.max(diff)) 0Hr每天发布大量与生活相关的资讯平台

打印输出 0Hr每天发布大量与生活相关的资讯平台

6.67572e-060Hr每天发布大量与生活相关的资讯平台

从结果中可以看到,两个模型文件的差异很小,为6.67572e-06 ,几乎可以忽略不计,所以这次转换的模型是可以直接应用的。0Hr每天发布大量与生活相关的资讯平台

  1. 转换后的模型需要注意输入格式,飞桨中输入格式需为NCHW格式。 此例中不涉及到输入中间层,如卷积层的输出,需要了解的是飞桨中的卷积层输出,卷积核的shape与TensorFlow有差异。
  2. 模型转换完后,检查转换前后模型的diff,需要测试得到的最大diff是否满足转换需求。
总结

X2Paddle提供了一个非常方便的转换方式,让大家可以直接将训练好的模型转换成Paddle Fluid版本。0Hr每天发布大量与生活相关的资讯平台

转换模型原先需要直接通过API对照表来重新实现代码。但是在实际生产过程中这么操作是很麻烦的,甚至还要进行二次开发。0Hr每天发布大量与生活相关的资讯平台

虽然飞桨相比其他AI平台上线较晚,但是凭借X2Paddle小工具,能快速将AI开发者吸引到自己的平台上来,后续的优势将愈加明显。0Hr每天发布大量与生活相关的资讯平台

除了本文提到的tensoflow2fluid,Paddle Fluid还支持caffe2fluid、onnx2fluid,大家可以根据自身的需求体验一下,有问题可以留言交流~0Hr每天发布大量与生活相关的资讯平台

  1. X2Paddle Github:https://github.com/PaddlePaddle/X2Paddle
  2. tensorflow2fluid: https://github.com/PaddlePaddle/X2Paddle/tree/master/tensorflow2fluid

ps:最后给大家推荐一个GPU福利 - Tesla V100免费算力!配合PaddleHub能让模型原地起飞~ 扫码下方二维码申请~0Hr每天发布大量与生活相关的资讯平台

也许你还喜欢

ups不间断电源原理分析

当前在电源市场上,所销售的电源类型较多,不同类型的电源,相应的功能与作用有所不同。在超

空间背景音乐怎么添加图文教程分享

怎样免费添加QQ空间背景音乐其实步骤也不多,很简单,主要有以下几个步骤,详情如下

万能转换器破解版怎么用?具体操作方

万能格式转换器是一款功能强大的格式转换软件,音频、视频、图片、光驱设备都可以通过万

rmvb转mp4格式转换器图文教程

在我们日常娱乐和日常工作中,现在网络上一些高清的视频一般都是rmvb格式的,而一些播放设

支付宝网络系统异常怎么解决

支付宝支付出现网络系统异常是因为IE浏览器设置Internet选项不恰当而导致的,去重新设置

win10蓝牙驱动破解版安装图文教程

win10蓝牙驱动怎么安装?请看下面具体操作方法。

微孔抛光镜面加工计数详解

一般来说,对于小孔微孔进行抛光,传统的加工方法可能会导致孔洞磨损,或者加工效果不理想。

短视频内容管理助手有哪些?怎么好用

随着短视频的普及,越来越多的人开始将大量的时间和精力投入到短视频的制作和分享中。因

模具设计学习知识分享

今天,我想谈谈我在模具设计CAD方面的一些心得体会。作为一个从事这个行业多年的工程师,

seo优化排名软件详细介绍

1. SEMrush SEMrush,一款广受好评的SEO神器,被誉为业内最佳的SEO分析工具。功能包括全