8

我正在尝试使用 cmake 构建我的 Android(本机)项目(将其从用于构建和运行良好的 gradle 实验性插件迁移)。

我有一些本机代码(将其称为“a”),它使用另一个外部预建库代码(将其称为“b”),我将两者链接如下:(根据https://developer.android.com/studio /projects/configure-cmake )

cmake_minimum_required(VERSION 3.4.1)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++14 -frtti -fno-common -fexceptions")

include_directories(
#a's include files' paths
#b's include files' paths)

file(GLOB_RECURSE A_SOURCES
#a's source files' paths)

add_library(a_lib SHARED ${A_SOURCES})
add_library(b_lib SHARED IMPORTED)
set_target_properties(b_lib PROPERTIES IMPORTED_LOCATION "b's .so path")
target_link_libraries(a_lib b_lib)

我通过了编译和链接步骤,android studio 继续在我的设备上安装 APK。但是,启动后,应用程序会在 logcat 中冻结,并显示以下内容:

E/ExceptionHandler: Uncaught Exception java.lang.UnsatisfiedLinkError: dlopen failed: library "libb_lib.so" not found
at java.lang.Runtime.loadLibrary0(Runtime.java:989)
at java.lang.System.loadLibrary(System.java:1530)
at com.mm.projectname.model.libloadingclassname.<clinit>(libloadingclassname.java:99)

我知道这是因为 b_lib 的 .so 不在 apk 中。我可以在 a 的共享库中看到 b 的符号和 a 的符号。

所以我的问题是我怎么能

  1. 要么将我的预建 b 的 .so 打包到 apk 中
  2. 或阻止系统在 libs 文件夹中查找 b.so 并使其仅在 a 的 .so 中查找 b 的符号。

我搜索了很多类似的帖子和问题(例如一个两个),但我什么也做不了。我真的在寻找正确的方法来做到这一点 - 这不会在未来产生问题(比如更改 targetSDKversion)。我还尝试使用最新的 ndk 版本构建预构建的库。

我可能犯了一个很小的错误,如果有人能指出这一点,我将不胜感激。

提前致谢

4

2 回答 2

2
  1. jniLibs在您的应用程序目录中创建一个文件夹main(例如:)/app/src/main
  2. 根据架构创建另一个文件夹。
  3. 将您的 .so 文件存储在此文件夹中。
于 2019-01-21T16:51:47.387 回答
1

有点晚了,但其他人可能会感兴趣(甚至是操作)。

我和操作员有同样的情况,我尝试了很多不同的事情。其中之一是尝试使用早期版本的 NDK 编译项目(我认为是 r14b,但我不确定)。我遇到了一个不同的错误,并设法找到了在“Invalid DT_NEEDED Entries”和“Missing SONAME”部分中详细描述的问题。此外,该问题已在此特定问题中进行了描述,并且已收到充分的答案。

如果您无法重新编译您正在使用的共享库以包含 SONAME,就像在我的情况下,您可以执行以下我所做的事情并设法工作:

  • 在项目树的 jniLibs 文件夹中包含该库,因此它被打包在您的 APK 中,但不要链接到它。
  • 使用此处描述的任何方法跟踪要合并的函数的符号。
  • 在 C++ 代码中创建相应函数的函数指针。
  • 在运行时加载共享库并映射到这些函数。

示例代码

标题

private:

typedef uint32_t (*InitX_t)();
typedef uint32_t (*DoX_t)();
typedef uint32_t (*GetX)(uint32_t, char*);

InitX_t InitX;
DoX_t DoX;
GetX_t GetX;

CPP

void *handle = dlopen("libMyLib.so", RTLD_NOW);

if(handle == nullptr)
{
    __android_log_print(ANDROID_LOG_INFO, "My Class", "Could not load library");
}
else
{
    __android_log_print(ANDROID_LOG_INFO, "My Class", "Library loaded");
}

InitX = (InitX_t)dlsym(handle, "InitX_SYMBOL");
ScanX = (ScanX_t)dlsym(handle, "ScanX_SYMBOL");
GetX = (GetX_t)dlsym(handle, "GetX_SYMBOL");

if(ScanX == nullptr || GetX == nullptr || InitX == nullptr)
{
    __android_log_print(ANDROID_LOG_INFO, "My Class", "Could not load functions.");
}

如果操作正确,您现在应该可以像往常一样使用这些功能了。不过,我知道,如果您是初学者,这并不是最简单的过程。

于 2018-09-26T15:47:17.890 回答