【问题标题】:Using Keras model for classification with Android在 Android 中使用 Keras 模型进行分类
【发布时间】:2019-11-30 19:39:35
【问题描述】:

我有一个训练有素的神经网络 (Alexnet) 作为 Keras 模型,应该用于 android 分类应用程序。首先,我将我的模型转换为 TensorFlow .pb 文件:

class Model2Graph(object):
    @staticmethod
    def Convert(ModelPath, OutputPath = "output"):

        print("[INFO] Converting model '{}' into TensorFlow graph...".format(ModelPath))

        # Get the model name
        ModelName = os.path.splitext(ModelPath)[0].rsplit(os.path.sep, 1)[-1]

        # Load the model from file
        Model = load_model(filepath = ModelPath)

        Backend.set_learning_phase(0)
        Session = Backend.get_session()

        OutputCount = len(Model.outputs)
        Temp = [None] * OutputCount
        NodeNames = [None] * OutputCount
        for i in range(OutputCount):
            NodeNames[i] = "output_node" + str(i)
            Temp[i] = tf.identity(Model.outputs[i], name = NodeNames[i])

        constant_graph = graph_util.convert_variables_to_constants(Session, Session.graph.as_graph_def(), NodeNames)    
        graph_io.write_graph(constant_graph, OutputPath, ModelName + ".pb", as_text = False)

此图(带有Labels.txt 文件)放置在我的 android 应用的 assets 文件夹中。

现在我将图像加载为 JPG 文件。我使用IrfanView 将其大小调整为227x227 图像(我的网络的输入大小并忽略纵横比)并使用我的应用程序加载此图像(图像称为1.jpg)。

应用开始对这张图片进行分类(这段代码基于TensorFlow示例):

public List<Float> Classify(Bitmap InputImage)
    {
        float[] Results = new float[_mClassLabels.size()];
        float[] Output = new float[_mWidth * _mHeight * 3];
        int[] intValues = new int[InputImage.getHeight() * InputImage.getWidth()];
        InputImage.getPixels(intValues, 0, InputImage.getWidth(), 0, 0, InputImage.getWidth(), InputImage.getHeight());

        for (int i = 0; i < intValues.length; ++i)
        {
            final int val = intValues[i];
            Output[i * 3] = ((val >> 16) & 0xFF) / 255.0f;
            Output[i * 3 + 1] = ((val >> 8) & 0xFF) / 255.0f;
            Output[i * 3 + 2] = (val & 0xFF) / 255.0f;
        }

        _mTensorFlowInterface.feed("conv2d_1_input", Output, 1L, _mWidth, _mHeight, 3);
        _mTensorFlowInterface.run(_mOutputName, false);
        _mTensorFlowInterface.fetch(_mOutputName[0], Results);

        // Convert the results into a list
        List<Float> Result = new ArrayList<Float>(Results.length);
        for(float f : Results)
        {
            Result.add(f);
        }

        return Result;
    }

我的 android 应用程序给了我以下分类结果:

  • 0.8430(猫)
  • 0.1569(狗)

但是当我使用我的 Python 应用程序和 Keras 模型时,我得到了不同的结果。 Python 应用程序给出以下结果:

  • 0.2255(猫)
  • 0.7744(狗)

现在我想知道为什么同一个网络会在两个应用程序中产生不同的结果。这里出了什么问题?

我的 Android / Java 不太好,所以我不确定预测代码是否正确。将 Keras 模型转换为 TensorFlow 的 .pb 文件也是如此。

【问题讨论】:

  • 我之前在 android 应用程序中使用过 tensorflow,对我来说,我需要将 .pb 文件转换为 .tflite 格式并运行推理。
  • 有趣。当您加载 .pb 文件时,您的应用程序是否会产生任何错误?因为我的 Android App 成功加载了 TensorFlow Graph。
  • 好吧,我从未尝试过加载 .pb 文件。我使用了 .tflite,它运行良好。我正在使用带有 mnist 数据集的 CNN 进行测试。似乎对于 android 开发,他们正在推动 tensorflow-lite 成为使用的核心框架。
  • 好的,谢谢。我会试试的。
  • 这种事情经常发生,因为 Python 代码在将图像输入图形之前会对图像进行预处理,而您的应用程序中的代码则不会。仅仅除以 255 可能还不够。

标签: python android tensorflow keras


【解决方案1】:

抱歉回复晚了。我为它创建了一个新帖子,因为我想分享解决方案。 感谢@Steven。您对 TensorFlow lite 的提示非常好。所以我使用

将 Keras 模型转换为 TensorFlow lite 图
Converter = lite.TFLiteConverter.from_keras_model_file(ModelPath)
open(OutputPath + os.path.sep + ModelName + ".tflite" , "wb") .write(Converter.convert())

现在可以使用 TensorFlow lite 加载此图:

private Interpreter _mTfLite;
private Interpreter.Options _mTfliteOptions = new Interpreter.Options();

private MappedByteBuffer loadModelFile(AssetManager Manager, String Path) throws IOException
{
    AssetFileDescriptor fileDescriptor = Manager.openFd(Path);
    FileInputStream inputStream = new 
    FileInputStream(fileDescriptor.getFileDescriptor());
    FileChannel fileChannel = inputStream.getChannel();
    long startOffset = fileDescriptor.getStartOffset();
    long declaredLength = fileDescriptor.getDeclaredLength();
    return fileChannel.map(FileChannel.MapMode.READ_ONLY, startOffset, declaredLength);
}

_mTfLite = new Interpreter(loadModelFile(_mAssetManager, Path), _mTfliteOptions.setNumThreads(1));

现在您可以在位图上运行预测:

private ByteBuffer _mInput;
private float[][] _mOutput;

public float[] Classify(Bitmap InputImage)
{
    _mInput.rewind();
    int width = InputImage.getWidth();
    int height = InputImage.getHeight();
    int[] pixels = new int[width * height];
    InputImage.getPixels(pixels, 0, width, 0, 0, width, height);

    int j = 0;
    for(int i = 0; i < pixels.length * 3; i = i + 3)
    {
        final int Pixel = pixels[j];

        _mInput.putFloat((((Pixel >> 16) & 0xFF)) / 1.0f);
        _mInput.putFloat((((Pixel >> 8) & 0xFF)) / 1.0f);
        _mInput.putFloat(((Pixel & 0xFF)) / 1.0f);

        j++;
    }

    _mTfLite.run(_mInput, _mOutput);

    return _mOutput[0];
}

但你需要添加

aaptOptions {
    noCompress "tflite"
    noCompress "lite"
}

到您的build.gradle 文件。否则,您的模型将在构建过程中被压缩,从而导致模型加载错误。

【讨论】:

  • 谢谢^^你是如何定义_mAssetManager的?而_mInput还没有初始化,怎么办?
  • 您可以在我的GitLab中查看代码的当前版本。
猜你喜欢
  • 1970-01-01
  • 2021-01-25
  • 2020-12-20
  • 2017-09-27
  • 1970-01-01
  • 1970-01-01
  • 2017-10-25
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多