我有一个 Android NDK 项目,它使用 Android.mk 文件构建和运行良好,它包含我作为共享库构建的本机代码(一个 C 文件),此代码依赖于第三方静态库(.a 文件) .
现在我正在尝试将其迁移到 Gradle。我目前的配置如下:
下的静态库Android.mk /static_libs folder
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := opus
LOCAL_SRC_FILES := lib/libopus.a
LOCAL_EXPORT_C_INCLUDES := $(LOCAL_PATH)/include
include $(PREBUILT_STATIC_LIBRARY)
主模块Android.mk文件:(在通常的jni/文件夹下)
LOCAL_PATH := $(call my-dir)
$(call import-add-path,$(LOCAL_PATH)/../static_libs)
include $(CLEAR_VARS)
LOCAL_MODULE:=opus_jni
LOCAL_SRC_FILES:= opus_jni.c
LOCAL_LDLIBS := -L$(SYSROOT)/usr/lib -llog -ldl
LOCAL_WHOLE_STATIC_LIBRARIES := opus
include $(BUILD_SHARED_LIBRARY)
$(call import-module,opus)
当我ndk-build
在主模块上运行时,唯一的输出是一个文件libopus_jni.so
,它是唯一用于构建最终 APK 的文件。
现在在 Gradle 上,我现在唯一想做的就是使用相同的预构建共享库和相同的代码创建一个 APK,所以我将生成的库复制到jniLibs/
文件夹中(我知道从 0.9 开始,Gradle 已经支持 JNI 库,如果文件放在这里)。项目构建,最终的 APK 确实包含.so
文件夹内的libs/
文件。 (我通过解压生成的APK进行了验证)
问题是,当我尝试使用任何本机方法时,都会出现以下错误:
05-28 14:57:53.393 3370-3370/com.opusgradle.app D/OB-OpusCodec﹕ testPcmToOpus()
05-28 14:57:53.393 3370-3370/com.opusgradle.app D/dalvikvm﹕ Trying to load lib /data/app-lib/com.opusgradle.app-2/libopus_jni.so 0x42630b18
05-28 14:57:53.393 3370-3370/com.opusgradle.app D/dalvikvm﹕ Added shared lib /data/app-lib/com.opusgradle.app-2/libopus_jni.so 0x42630b18
05-28 14:57:53.393 3370-3370/com.opusgradle.app D/OB-OpusCodec﹕ Trying to initialize...
05-28 14:57:53.393 3370-3370/com.opusgradle.app W/dalvikvm﹕ No implementation found for native Lcom/opusgradle/app/OpusCodec;.initOpusEncoder:(II)V
05-28 14:57:53.393 3370-3370/com.opusgradle.app D/AndroidRuntime﹕ Shutting down VM
05-28 14:57:53.393 3370-3370/com.opusgradle.app W/dalvikvm﹕ threadid=1: thread exiting with uncaught exception (group=0x415e0ba8)
05-28 14:57:53.393 3370-3370/com.opusgradle.app E/AndroidRuntime﹕ FATAL EXCEPTION: main
Process: com.opusgradle.app, PID: 3370
java.lang.UnsatisfiedLinkError: Native method not found: com.opusgradle.app.OpusCodec.initOpusEncoder:(II)V
at com.opusgradle.app.OpusCodec.initOpusEncoder(Native Method)
at com.opusgradle.app.OpusCodec.(OpusCodec.java:23)
at com.opusgradle.app.MainActivity.testPcmToOpus(MainActivity.java:78)
at com.opusgradle.app.MainActivity.access$000(MainActivity.java:22)
at com.opusgradle.app.MainActivity$1.onClick(MainActivity.java:64)
at android.view.View.performClick(View.java:4438)
at android.view.View$PerformClick.run(View.java:18422)
at android.os.Handler.handleCallback(Handler.java:733)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:136)
at android.app.ActivityThread.main(ActivityThread.java:5017)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
at dalvik.system.NativeStart.main(Native Method)
如您所见,.so
文件已找到并加载,但对本机函数的调用失败。Java 代码与原始项目中的代码相同,运行良好,基本上对库的调用如下所示:
static
{
System.loadLibrary("opus_jni");
}
似乎当需要静态库时,Gradle 需要将其他内容复制到jniLibs文件夹中,我尝试将 .a 文件也复制到该文件jniLibs/
夹中,但没有任何成功。
对于在 Gradle 上正确构建具有静态库依赖项的预构建共享库还有其他要求吗?