Android NDK导入模块/代码重用

Gra*_*eme 12 java-native-interface android-ndk import-module

早上!

我创建了一个小型NDK项目,它允许通过JNI在Java和C++之间动态序列化对象.逻辑的工作原理如下:

Bean - > JavaCInterface.Java - > JavaCInterface.cpp - > JavaCInterface.java - > Bean

问题是我想在其他项目中使用此功能.我从项目中分离出测试代码并创建了一个"Tester"项目.测试人员项目将Java对象发送到C++,然后将其回送到Java层.

我认为链接非常简单 - (NDK/JNI的"简单"通常是令人沮丧的一天)我将JNIBridge项目添加为源项目,并将以下行添加到Android.mk:

NDK_MODULE_PATH=.../JNIBridge/jni/"

JNIBridge/JNI/JavaCInterface/Android.mk:

...
include $(BUILD_STATIC_LIBRARY)
Run Code Online (Sandbox Code Playgroud)

JNITester/JNI/Android.mk:

...
include $(BUILD_SHARED_LIBRARY)
$(call import-module, JavaCInterface)
Run Code Online (Sandbox Code Playgroud)

一切正常.依赖JavaCInterface模块头文件的C++文件工作正常.此外,Java类可以愉快地使用JNIBridge项目中的接口.所有的链接都很开心.

不幸的是,包含本机方法调用的JavaCInterface.java无法看到位于静态库中的JNI方法.(从逻辑上讲,它们位于同一个项目中,但都通过上述机制导入到您希望使用它们的项目中).

我目前的解决方案如下.我希望有人可以建议一些能够保持我正在努力实现的模块性的东西:


我目前的解决方案是在调用项目中包含JavaCInterface cpp文件,如下所示:

LOCAL_SRC_FILES := FunctionTable.cpp $(PATH_TO_SHARED_PROJECT)/JavaCInterface.cpp
Run Code Online (Sandbox Code Playgroud)

但我宁愿不这样做,因为如果我改变了JavaCInterface架构,我会需要更新每个依赖项目.


我可以在每个本地项目中创建一组新的JNI方法签名,然后链接到导入的模块.同样,这会严格限制实现.

Gra*_*eme 15

经过多次血汗和泪水,我已经想到了这一点.

  • Android JNI从SHARED_LIBRARY唯一加载它的二进制文件.
  • JNI将尝试将本机调用链接到加载的共享库中的相应方法签名/存根(它不会查看链接的共享库).
  • 您可以使用这些方法创建静态库,并将其构建到应用程序使用的共享库中.

您可以使用Andriod.xml中的以下代码在原始项目中构建静态库:

include $(CLEAR_VARS)
LOCAL_CFLAGS    := -O0
LOCAL_MODULE    := LibraryToBeUsedInsideSharedLib
LOCAL_SRC_FILES := ...
include $(BUILD_STATIC_LIBRARY) // This builds a "Static Object" here:
                                // /Project/obj/local/armeabi/libLibraryToBeUsedInsideSharedLib.a

include $(CLEAR_VARS)
LOCAL_MODULE       := LibraryCalledFromJava
LOCAL_SRC_FILES    := ...
LOCAL_STATIC_LIBRARIES := LibraryToBeUsedInsideSharedLib
include $(BUILD_SHARED_LIBRARY)
Run Code Online (Sandbox Code Playgroud)

LOCAL_STATIC_LIBRARIES包括共享库中的静态库.在您的Java代码中,您现在可以调用它:

System.loadLibrary("LibraryCalledFromJava");
Run Code Online (Sandbox Code Playgroud)

您应该能够LibraryToBeUsedInsideSharedLib从java代码中的任何位置调用库中的任何本机方法.

您可以通过将该libLibraryToBeUsedInsideSharedLib.a文件添加到外部项目的Android.xml 来导出该文件并将其用于其他项目:

include $(CLEAR_VARS)
LOCAL_MODULE            := LibraryToBeUsedInsideSharedLib
LOCAL_LDLIBS            := -llog/
LOCAL_SRC_FILES         := $(MY_PREBUILT_LIB_DIR)/libLibraryToBeUsedInsideSharedLib.a
include $(PREBUILT_STATIC_LIBRARY)
Run Code Online (Sandbox Code Playgroud)