Android如何使用libjpeg-turbo库

标签 android android-ndk

多亏了这个,我终于成功地构建了 libjpeg-turbo 静态库 libjpeg-turbo for android 现在我有一个 libjpeg.a 和一个由 ndk-build 生成的 libsimd.a

但是我还没有找到任何关于下一步该做什么的信息?我正在使用 BitmapFactory 中的构建将 jpeg 从缓冲区(从套接字)解码为位图

byte[] jpgBits = new byte[jpgBitsLen];
dis.readFully(jpgBits);
Bitmap bmp = BitmapFactory.decodeByteArray(jpgBits, 0, jpgBitsLen);

我如何用 libjpeg-turbo 替换 BitmapFactory.decodeByteArray?

我用这个在我的电脑上编码流

tjhandle rmfdJpegCompressor = tjInitCompress();
tjCompress2(rmfdJpegCompressor, (unsigned char *)s_rmfdPixels, MFD_WH, 0, MFD_WH, TJPF_BGRX,
            &rmfdBits, &rmfdBitsLen, TJSAMP_420, RMFD_JPEG_QUALITY,
            0);
tjDestroy(rmfdJpegCompressor);

哪个工作正常,所以我认为必须有一个 android 等效项?

我读过这个 https://wiki.linaro.org/BenjaminGaignard/libjpeg-turboAndSkia 这是否意味着使用它的唯一方法是重建 android 源以便它使用 libjpeg-turbo?我在某处读到有一个兼容性 api 和一个用于 libjpeg-turbo 的 native api,我很乐意使用任何最简单的 api,因为我不喜欢重建 android

我试过以下方法: 在我的项目根目录下,我创建了文件夹 jni/include 并将 turbojpeg.h 放在那里 在我的项目根目录下,我创建了文件夹 jni/prebuilt 并将 libjpeg.a 放在那里

在我的java代码中,我把

private native int tjInitDecompress(); 

在主 Activity 中 在 onCreate 中我添加

int i = tjInitDecompress();
Log.d("MainActivity", "i="+i);

它构建并运行但在 tjInitDecompress 时崩溃

在日志中它说: 未找到原生 Lcom/example.jpegtest/MainActivity;.tjInitDecompress ()I 的实现

谢谢

最佳答案

好吧,这是一大堆工作,但我终于有了一些工作,所以我想让任何感兴趣的人知道我是如何做到的。

首先,我按照此处所述构建了 hello-jin 演示 https://developer.android.com/tools/sdk/ndk/index.html

然后我创建了一个新项目,复制了 jni 并更改了 c func 的名称以匹配新的包和类名称。不要在你的包名中使用 - 和 _ 否则你会遇到问题。最好是 a-x0-9。

然后我将所有 libjpeg-turbo 文件和目录复制到 jni 中并测试 ndk-build 是否仍然有效

然后我像这样为 libjpg 函数创建了一个 jni 包装器 tjpegini-arm.c

/*
 * Copyright (C) 2009 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 */
#include <jni.h>
#include "turbojpeg.h"


/*
 * Class:     libjpegturbo_jniapi
 * Method:    tjInitDecompress
 * Signature: ()I
 */
//package com.design2112.fbmslpit
//public class MainActivity
jint JNICALL Java_com_design2112_fbmslpit_MainActivity_tjInitDecompress
  (JNIEnv *env, jobject thisObj)
{
    return (int)tjInitDecompress();
}


/*
 * Class:     libjpegturbo_jniapi
 * Method:    tjDecompressHeader2
 * Signature: (I[BI)I
 */
jint JNICALL Java_com_design2112_fbmslpit_MainActivity_tjDecompressHeader2
  (JNIEnv *env, jobject thisObj, jint handle, jbyteArray jpegBuf, jint jpegSize)
{
    jbyte *real_jpegBuf = (*env)->GetByteArrayElements(env, jpegBuf, 0);
    if (!real_jpegBuf) return -1;
    //jsize length = (*env)->GetArrayLength(env, real_jpegBuf);

    /*for (i = 0; i < length; i++) {
        sum += inCArray[i];
    }*/

    int width, height, jpegSubsamp;
    int ret =  tjDecompressHeader2((tjhandle)handle,
                (unsigned char *)real_jpegBuf, (unsigned long)jpegSize, &width, &height,
                &jpegSubsamp);
    if(ret!=0) {
        return 0;
    }

    // ok, so pack width and height together
    return width<<16 | height;
}

/*
 * Class:     libjpegturbo_jniapi
 * Method:    tjDecompress2
 * Signature: (I[BI[IIIIII)V
 */
void JNICALL Java_com_design2112_fbmslpit_MainActivity_tjDecompress2
  (JNIEnv *env, jobject thisObj, jint handle, jbyteArray jpegBuf, jint jpegSize, jintArray dstBuf,
  jint width, jint pitch, jint height, jint pixelFormat, jint flags)
{
    jbyte *real_jpegBuf = (*env)->GetByteArrayElements(env, jpegBuf, 0);
    if (!real_jpegBuf) return;
    jint *real_dstBuf = (*env)->GetIntArrayElements(env, dstBuf, 0);
    if (!real_dstBuf) return;

    jsize length = (*env)->GetArrayLength(env, jpegBuf);
    tjDecompress2((tjhandle)handle,
                (unsigned char *)real_jpegBuf, (unsigned long)jpegSize, (unsigned char *)real_dstBuf,
                 width, pitch, height, pixelFormat, flags);
}

/*
 * Class:     libjpegturbo_jniapi
 * Method:    tjDestroy
 * Signature: (I)V
 */
void JNICALL Java_com_design2112_fbmslpit_MainActivity_tjDestroy
  (JNIEnv *env, jobject thisObj, jint handle)
{
    tjDestroy((tjhandle)handle);
}

重要,您必须将 com_design2112_fbmslpit_MainActivity 重命名为您的包和类才能正常工作

将 tjpegini-arm.c 添加到 Android.mk 生成文件,然后在 jni 目录中运行 ndk-build

ndk-build NDK_PROJECT_PATH=. APP_BUILD_SCRIPT=./Android.mk obj/local/armeabi/libjpeg.so  LOCAL_ARM_MODE=arm

并将.so复制到正确的名称和位置

cp obj/local/armeabi/libjpeg.so ../libs/armeabi/libtjpegjni-arm.so

然后在我的 MainAvtivity.java 中

public class MainActivity extends Activity {


    public native int tjInitDecompress();
    public native int tjDecompressHeader2(int handle, byte[] jpegBits, int jpegBitsLen);    
    public native void tjDecompress2(int handle, byte[] jpegBits,
            int jpegBitsLen, int[] outbuffer, int width, int pitch, int height,
            int pixelFormat, int flags);
    public native void tjDestroy(int handle);

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);


        setContentView(R.layout.activity_main);

        if (savedInstanceState == null) {
            getFragmentManager().beginTransaction()
                    .add(R.id.container, new PlaceholderFragment())
                    .commit();
        }

        File sdcard = Environment.getExternalStorageDirectory();

        //Get the text file
        File file = new File(sdcard,"/Download/test.jpg");
        int jpegBitsLen = (int) file.length();
        byte[] jpegBits = new byte[jpegBitsLen];
        DataInputStream dis;
        try {
            dis = new DataInputStream(new FileInputStream(file));
            dis.readFully(jpegBits);
            dis.close();
        } catch (FileNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

        System.loadLibrary("tjpegjni-arm");

        int jpegDec = tjInitDecompress();

        int wh = tjDecompressHeader2(jpegDec, jpegBits, jpegBitsLen);
        int width = wh>>16;
        int height = wh&0x7fff;

        int[] buffer = new int[width*height];
        tjDecompress2(jpegDec, jpegBits, jpegBitsLen, buffer, width, 0/*pitch*/, height, 2 /*TJPF_RGBX*/, 0);

        tjDestroy(jpegDec);

        Bitmap bmp = Bitmap.createBitmap(buffer, width, height, Bitmap.Config.ARGB_8888);

    }

基本上就是这样。你可以以任何你想要的方式显示 bmp。

这对我来说也是一大堆工作,因为我根本没有 jni ndk 经验。如果有人觉得这有用,请给我发电子邮件。

更新,这是一个令人震惊的消息,解码 450x450 图像需要 20 毫秒。 内置的 BitmapFactory.decodeByteArray 的作用大致相同!

如果有其他人尝试此操作并得到不同的结果,请记下

关于Android如何使用libjpeg-turbo库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24548952/

相关文章:

android - 对于显示 UnsatisfiedLinkError : No implementation found for . stringFromJNI2() 的第二个 native 函数

java - Android - java.lang.RuntimeException :App is crashing as soon as it launches

android - java.lang.NoClassDefFoundError : Failed resolution of: Ljava/time/LocalTime; 错误

java - 空指针异常 Android 应用程序

android - 让 Android 设备成为 C 游戏 Remote 的最佳方法

java - 关于在 JNI 中分配的数组

android - 如何在android中的Viewpager中添加imageURL。我需要一个平滑的滑动 Action

c++ - 无法构建 opengl es 2.0 c++ native android 项目。 ( fatal error : android/sensor. h:没有这样的文件或目录)

java - 示例 hello-jni 上的 Android NDK java.lang.UnsatisfiedLinkError

android - 使用 WebView 进行 OAuth2 授权