TensorFlow集成Android工程的框架

欢迎Follow我的GitHub,关注我的简书

在Android工程中,集成TensorFlow模型。运行TensorFlow的默认Android工程,请参考

Android源码:https://github.com/SpikeKing/TFAndroid/tree/master

库及模型的大小

libtensorflow_inference.so  10.2 M
libandroid_tensorflow_inference_java.jar  27 KB
optimized_tfdroid.pb  291 B

如果将so转换为jar库,参考,则TF的so由10.2M缩小至4.1M。

TF Android

TensorFlow

TF模型源码:
https://github.com/SpikeKing/MachineLearningTutorial/blob/master/tests/android_test.py

创建TensorFlow模型,简单的y=WX+b,存储图信息write_graph,存储参数信息saver.save。输入数据placeholder是I,输出数据是O

import tensorflow as tf

I = tf.placeholder(tf.float32, shape=[None, 3], name='I')  # input
W = tf.Variable(tf.zeros(shape=[3, 2]), dtype=tf.float32, name='W')  # weights
b = tf.Variable(tf.zeros(shape=[2]), dtype=tf.float32, name='b')  # biases
O = tf.nn.relu(tf.matmul(I, W) + b, name='O')  # activation / output

saver = tf.train.Saver()
init_op = tf.global_variables_initializer()

with tf.Session() as sess:
    sess.run(init_op)

    tf.train.write_graph(sess.graph_def, './data/android/', 'tfdroid.pbtxt')  # 存储TensorFlow的图

    # 训练数据,本例直接赋值
    sess.run(tf.assign(W, [[1, 2], [4, 5], [7, 8]]))
    sess.run(tf.assign(b, [1, 1]))

    # 存储checkpoint文件,即参数信息
    saver.save(sess, './data/android/tfdroid.ckpt')

创建Freeze的图,将图结构与参数组合在一起,生成模型,参考

def gnr_freeze_graph(input_graph, input_saver, input_binary, input_checkpoint,
                     output_node_names, output_graph, clear_devices):
    """
    将输入图与参数结合在一起
    
    :param input_graph: 输入图
    :param input_saver: Saver解析器
    :param input_binary: 输入图的格式,false是文本,true是二进制
    :param input_checkpoint: checkpoint,检查点文件
    
    :param output_node_names: 输出节点名称
    :param output_graph: 保存输出文件
    :param clear_devices: 清除训练设备
    :return: NULL
    """
    restore_op_name = "save/restore_all"
    filename_tensor_name = "save/Const:0"

    freeze_graph.freeze_graph(
        input_graph=input_graph,  # 输入图
        input_saver=input_saver,  # Saver解析器
        input_binary=input_binary,  # 输入图的格式,false是文本,true是二进制
        input_checkpoint=input_checkpoint,  # checkpoint,检查点文件
        output_node_names=output_node_names,  # 输出节点名称
        restore_op_name=restore_op_name,  # 从模型恢复节点的名字
        filename_tensor_name=filename_tensor_name,  # tensor名称
        output_graph=output_graph,  # 保存输出文件
        clear_devices=clear_devices,  # 清除训练设备
        initializer_nodes="")  # 初始化节点

优化模型,剪切节点,模型只保留输入输出的参数。

def gnr_optimize_graph(graph_path, optimized_graph_path):
    """
    优化图
    :param graph_path: 原始图
    :param optimized_graph_path: 优化的图
    :return: NULL
    """
    input_graph_def = tf.GraphDef()  # 读取原始图
    with tf.gfile.Open(graph_path, "r") as f:
        data = f.read()
        input_graph_def.ParseFromString(data)

    # 设置输入输出节点,剪切分支,大约节省1/4
    output_graph_def = optimize_for_inference_lib.optimize_for_inference(
        input_graph_def,
        ["I"],  # an array of the input node(s)
        ["O"],  # an array of output nodes
        tf.float32.as_datatype_enum)

    # 存储优化的图
    f = tf.gfile.FastGFile(optimized_graph_path, "w")
    f.write(output_graph_def.SerializeToString())

执行函数,生成模型,frozen_tfdroid.pboptimized_tfdroid.pb

if __name__ == "__main__":
    input_graph_path = MODEL_FOLDER + MODEL_NAME + '.pbtxt'  # 输入图
    checkpoint_path = MODEL_FOLDER + MODEL_NAME + '.ckpt'  # 输入参数
    output_path = MODEL_FOLDER + 'frozen_' + MODEL_NAME + '.pb'  # Freeze模型

    gnr_freeze_graph(input_graph=input_graph_path, input_saver="",
                     input_binary=False, input_checkpoint=checkpoint_path,
                     output_node_names="O", output_graph=output_path, clear_devices=True)

    optimized_output_graph = MODEL_FOLDER + 'optimized_' + MODEL_NAME + '.pb'

    gnr_optimize_graph(output_path, optimized_output_graph)

Android

编译Android的库,参考,或者,直接在Nightly中下载,参考,archive.zip,大约158M。

创建Android工程,添加app/libs/中添加库文件。

armeabi-v7a/libtensorflow_inference.so
libandroid_tensorflow_inference_java.jar

在build.gradle中,添加

android {
    sourceSets {
        main {
            jniLibs.srcDirs = ['libs']
        }
    }
}

在app/src/main/assets中,添加模型optimized_tfdroid.pb文件。

在MainActivity中,添加so库。

static {
    System.loadLibrary("tensorflow_inference");
}

模型文件在assets中,TF的核心接口类TensorFlowInferenceInterface。

private static final String MODEL_FILE = "file:///android_asset/optimized_tfdroid.pb";

private TensorFlowInferenceInterface mInferenceInterface;

初始模型文件

mInferenceInterface = new TensorFlowInferenceInterface();
mInferenceInterface.initializeTensorFlow(getAssets(), MODEL_FILE);

模型Feed数据,输入点名称是INPUT_NODE,输入结构INPUT_SIZE,输入数据inputFloats。

float[] inputFloats = {num1, num2, num3};
mInferenceInterface.fillNodeFloat(INPUT_NODE, INPUT_SIZE, inputFloats);

模型执行文件,输出点名称是OUTPUT_NODE,即"O"

mInferenceInterface.runInference(new String[]{OUTPUT_NODE});

输出数据结构

float[] resu = {0, 0};
mInferenceInterface.readNodeFloat(OUTPUT_NODE, resu);

最后,在layout中创建GUI布局。

效果

Demo

TensorFlow集成至春雨医生

CY-TF

That's all! Enjoy it!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,046评论 25 707
  • 曾国藩把学习作为提高个人修养的基本要求,将“敬德修业”当做自己进步的基础。在他看来,要想取得事业的成功,就必须要积...
    friend6522阅读 1,026评论 0 0
  • 这篇文章是本人在阅读Dozer官方文档(5.5.1版本,官网已经一年多没更新了)的过程中,整理下来我认为比较基础的...
    静默虚空阅读 15,362评论 2 13
  • 原帖:http://www.cnblogs.com/kym/archive/2012/05/14/2498728....
    iFavorite阅读 610评论 0 0
  • 那一曲恋歌,吟唱多年 您听过鲤鱼跳跃的声音吗?这是清晨的易湖湖畔,请你侧耳倾听。多少人一生没有听过这种声音,住在湖...
    躲开让我来阅读 84评论 0 0