繁体   English   中英

在Android Studio中使用预建的共享库

[英]Using Pre-built Shared Library in Android Studio

我需要在我的Android项目中使用自定义的预构建共享库(基于独立ndk构建为libdynamic.so)。 我在路径src / main中创建了一个文件夹“ jniLibs”,然后在其中创建了4个文件夹,即“ armeabi”,“ armeabi-v7a”,“ x86”,“ x86_64”。 我已经将预构建的库放在所有这4个文件夹中。

现在,从我的本机代码中,我想调用该库的函数。 以以下方式(包含在cmakelists.txt中的标头):

extern "C"
JNIEXPORT jstring JNICALL
Java_demo_co_ru_jnilibtest_MainActivity_stringFromJNI(
    JNIEnv *env,
    jobject /* this */) {

     float inv = rsqrt(3);  //FUNCTION FROM LIBRARY (libdynamic.so)

     std::string hello = "Hello ";
     return env->NewStringUTF(hello.c_str());
}

我收到以下错误:

  1. Error:error: cannot find -ldynamic

  2. Error:(19) undefined reference to 'rsqrt(float)'

  3. Error:error: linker command failed with exit code 1 (use -v to see invocation)

似乎共享库未找到。 我在CMakeLists.txt中输入了以下值

include_directories( src/main/cpp/include) #include header of libdynamic.so
target_link_libraries(native-lib dynamic)  #dependency of native-lib on libdynamic.so

我在gradle build(app)中添加了以下其他条目:

defaultConfig {
  ndk{
        abiFilters 'armeabi', 'armeabi-v7a', 'x86', 'x86_64'
    }
}

sourceSets {
    main {
        jni.srcDirs = ['src/main/jni', 'src/main/jniLibs/']
    }
}


externalNativeBuild {
    cmake {
        path "CMakeLists.txt"
    }
}

我能够使用android push和android shell成功运行该库。 这是使用Android Studio的apk构建引起的。 我正在使用Android Studio版本2.3.3。 非常感谢您的帮助。

为了在Android环境中使用CMake加载您的库,您将必须在本地库CMakeLists.txt中添加以下代码:

设置你的libs路径

set(LIBS_DIR ${CMAKE_SOURCE_DIR}/../jniLibs)

添加要导入的库

add_library(DYNAMIC_LIB SHARED IMPORTED)

设置每个ABI的库位置

set_target_properties(DYNAMIC_LIB PROPERTIES IMPORTED_LOCATION ${LIBS_DIR}/${ANDROID_ABI}/lidynamic.so)

将导入的库链接到目标

target_link_libraries(native-lib DYNAMIC_LIB)

并在本地库build.gradle中:

defaultConfig{
    ...
    externalNativeBuild{
        // Specify the toolchain which was used to cross compile libdynamic.so because Android Studio assumes that you used clang
        arguments '-DANDROID_TOOLCHAIN=gcc'
    }
}

使用-L将sysroot lib dir添加到LDFLAGS,因为如果我没记错的话,libdynamic也依赖libc,libdl,并且至少应该需要libm。

路径应为:

$ NDK /平台/ android-(平台版本)/ arch-(体系结构)/ usr / lib

我能够使用Android.mk而不是cmake使其工作。 我发布Android.mk的配置和内容以及gradle构建,以防万一有人需要它。

在“ app”下创建一个文件夹“ jni”。 创建另一个自定义文件夹“ yourlibs”,并将所有预先构建的库放在相应的“ TARGET_ARCH_ABI”文件夹中的“ yourlibs”文件夹中。 例如,在我的情况下:

  • jni / yourlibs / armeabi / libdynamic.so
  • jni / yourlibs / armeabi-v7a / libdynamic.so
  • jni / yourlibs / x86 / libdynamic.so
  • jni / yourlibs / x86_64 / libdynamic.so

现在,请按照下列步骤操作:

  1. 在“ jni”文件夹中创建一个“ C”文件,从中可以调用“ libdynamic.so”中定义的函数。 将必要的头文件添加到您创建的“ C”文件中。 对我来说,它是“ uselib.c”和“ header.h”
  2. 在“ jni”文件夹中创建一个名为“ Android.mk”的文件

在Android.mk中添加以下内容

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES := yourlibs/$(TARGET_ARCH_ABI)/libdynamic.so
LOCAL_MODULE := add_prebuilt
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_SRC_FILES  := uselib.c
LOCAL_MODULE     :=  use-lib
LOCAL_SHARED_LIBRARIES := add_prebuilt
include $(BUILD_SHARED_LIBRARY)

更新gradle build(app)文件以使用“ Android.mk”而不是cmake:

里面的“ android => defaultConfig”

 ndk{
        abiFilters 'armeabi', 'armeabi-v7a', 'x86', 'x86_64'
    }

在“ android”里面

externalNativeBuild {
    ndkBuild {
        path "jni/Android.mk"
    }
}

这将创建一个名为“ use-lib”的库,该库使用“ libdynamic.so”,并将这两个库都打包到apk的lib文件夹中。 您可以使用apk分析器(Android Studio => Build => Analyze Apk ...)进行检查。 要使用“ use-lib”,请使用jni调用,例如:

static {
    System.loadLibrary("use-lib");
}

public native String stringFromJNI(); 

注意:我从C代码中删除了extern“ C”语句。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM