Android NDK导入模块/代码重用

时间:2011-06-15 08:35:47

标签: java-native-interface android-ndk import-module

早!

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

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

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

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

NDK_MODULE_PATH=.../JNIBridge/jni/"

JNIBridge / JNI / JavaCInterface / Android.mk:

...
include $(BUILD_STATIC_LIBRARY)

JNITester / JNI / Android.mk:

...
include $(BUILD_SHARED_LIBRARY)
$(call import-module, JavaCInterface)

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

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

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


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

LOCAL_SRC_FILES := FunctionTable.cpp $(PATH_TO_SHARED_PROJECT)/JavaCInterface.cpp

但我宁愿不这样做,因为如果我更改了JavaCInterface体系结构,我将需要更新每个依赖项目。


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

1 个答案:

答案 0 :(得分: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)

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

System.loadLibrary("LibraryCalledFromJava");

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

您可以导出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)