Android NDK-错误链接共享库和JNI包装器

时间:2019-01-31 20:17:51

标签: android c android-ndk java-native-interface

我试图将使用NDK-Standalone工具链生成的共享库链接到另一台构建机上。然后使用特定的.so将其放到Android Studio中。从那里,我使用javah创建了一个jni.h文件,然后帮助我为函数调用编写了.c JNI。

其次这个例子How do I compile any native (C, C++) library using NDK in the form of shared libraries

ndk-build确实可以编译并且似乎可以正常工作,但是尝试在手机上运行该应用程序时,我在

处收到错误消息
static { 
    System.loadLibrary("testLib")
}

testLib.so即使在libs/armeabi-v7a/testLib.so目录中也找不到它

当前错误:

 01-31 14:41:53.779 19024-19024/com.jolopy.testing_02 E/AndroidRuntime: FATAL EXCEPTION: main
        Process: com.jolopy.testing_02, PID: 19024
        java.lang.UnsatisfiedLinkError: dalvik.system.PathClassLoader[DexPathList[[zip file "/data/app/com.jolopy.testing_02-2/base.apk"],nativeLibraryDirectories=[/data/app/com.jolopy.testing_02-2/lib/arm64, /vendor/lib64, /system/lib64]]] couldn't find "testLib.so"
            at java.lang.Runtime.loadLibrary(Runtime.java:367)
            at java.lang.System.loadLibrary(System.java:1076)
            at com.jolopy.testing_02.TestLib.<clinit>(TestLib.java:6)
            at com.jolopy.testing_02.MainActivity.onCreate(MainActivity.java:18)

如何构建的.so文件:

arm-linux-androideabi-gcc -c -fPIC testLib.c -o test.o

arm-linux-androideabi-gcc test.o -o testing.so

从那里,我使用javah编写了一个JNI包装器类,该类生成了testing_Android.h文件。从那里产生的testing_Android.c JNI封装,我使用从我的testLib.c库调用函数:

#include "testLib.h"
//Including Machine Generated Header
#include "testing_Android.h"
#include <stdio.h>

JNIEXPORT void JNICALL Java_com_jolopy_testing_102_TestLib_testinglib_1Initialize
  (JNIEnv *env, jobject obj){
    (void)env;
    (void)obj;

    testing_Initialize();
}


JNIEXPORT jint JNICALL Java_com_jolopy_testing_102_TestLib_testinglib_1Get_1Count
  (JNIEnv *env, jobject obj){
    (void)env;
    (void)obj;

    return(testing_Get_Count());
}


JNIEXPORT jint JNICALL Java_com_jolopy_testing_102_TestLib_testinglib_1Get_1CurrentName
  (JNIEnv *env, jobject obj, jlong ptr, jint x){
    (void)env;
    (void)obj;

    return (testing_Get_CurrentName((char *)ptr , (int)x));

}

从那里,我在Android的jni文件夹中有5个文件,这是我从以下位置运行ndk-build命令的地方:

 testing.so | testing_Android.h | testing_Android.c | Application.mk | Android.mk

Android.mk:

LOCAL_PATH :=$(call my-dir)
include $(CLEAR_VARS)

LOCAL_MODULE := testLib
LOCAL_SRC_FILES := testing.so
LOCAL_EXPORT_C_INCLUDES := testing_Android.c
include $(PREBUILT_SHARED_LIBRARY)

Application.mk:

APP_PLATFORM := android-19
APP_ABI := armeabi-v7a

在我的进步有什么建议或故障,您可能会看到,我不将不胜感激。

-Cheers!

2 个答案:

答案 0 :(得分:1)

在下面更改

APP_ABI := armeabi-v7a 

APP_ABI := arm64-v8a

由于错误日志的原因,您的设备为arm64 ABI。


如果您要开始一个新的Android NDK项目,我建议您从 Android Studio + CMake 工具链开始,请参见此处,了解我基于Android Studio和CMake的个人JniExample项目:< / p>

答案 1 :(得分:0)

我没有使用ndk独立工具链,而是在构建计算机上使用了ndk-build,然后生成了我需要的libs文件。从那里,我添加了libs文件,其内容为:

  1. arm64-v8a
  2. armeabi-v7a
  3. x86
  4. x86_64

进入src> main> jniLibs下的Android目录,然后存储所有文件,并将命令添加到我的应用程序build.gradle中:

NOT NULL constraint failed: app_my_model.user_id

要使ndk-build正常工作,我必须包括jniWrapper.c文件。

Android.mk:

sourceSets.main{
        jniLibs.srcDir 'src/main/jniLibs'
    }

Application.mk(我正在测试“全部”,但不需要):

LOCAL_PATH :=$(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := testing
LOCAL_SRC_FILES := jniWrapper.c \
                    testLib.c
include $(BUILD_SHARED_LIBRARY)

我知道这是一个解决方案,但是由于某种原因,我无法弄清楚该工具链是否可以正常工作。