繁体   English   中英

Android:如何使用JNI代码中的现有C ++静态库?

[英]Android: How to use an existing C++ static library from your JNI code?

我见过与此问题类似的问题,但情况并不完全相同,也无法获得适用于我的问题的答案。

我有C ++库的源代码。 我们需要将此库用作android应用程序的一部分,但也需要第三方将其用作C ++库。

我有一个makefile,它使用ndk的编译器从库的源代码中生成.a文件。 那就是纯C ++部分。

在Java方面,我有一个简单的演示项目,其中包含一个包含按钮的简单活动。 当按下按钮时,将调用本机代码。

只要我不尝试从JNI函数从库中调用函数,一切就可以正常工作。

以下是该库的来源:

SimpleMath.h

int Add(int aNumber1, int aNumberB);

SimpleMath.cpp

#include "SimpleMath.h"

int Add(int aNumberA, int aNumberB)
{
  return aNumberA + aNumberB;
}

生成文件

APP      = simple_app
LIBRARY  = simple_library.a
OBJECTS  = SimpleMath.o
CFLAGS   = -Wall -pedantic
NDK_PATH = /home/jug/perforce/jug_navui_personal_main/Env/Linux/Android/ndk/r7c
CXX      = $(NDK_PATH)/toolchains/arm-linux-androideabi-4.4.3/prebuilt/linux-x86/bin/arm-linux-androideabi-g++
AR       = $(NDK_PATH)/toolchains/arm-linux-androideabi-4.4.3/prebuilt/linux-x86/bin/arm-linux-androideabi-ar
SYSTEM_LIBS = -lstdc++ -lm
INCLUDE_PATH += ${NDK_PATH}/platforms/android-9/arch-arm/usr/include

all: $(LIBRARY)

$(LIBRARY): 
    $(CXX) -c SimpleMath.c
    $(AR) rcs simple_library.a SimpleMath.o

clean:
    rm *.o *.a

在Java方面,这些是文件:

你好,jni.c

#include <string.h>
#include <jni.h>

#include "../../../native/simple_library/SimpleMath.h"

jstring Java_com_amstapps_samples_draft08jni_MainActivity_helloJni(JNIEnv* env, jobject obj)
{
    // Uncommenting the line below results in undefined-symbol compile error
    //int d = Add(1, 2);

    return (*env)->NewStringUTF(env, "Hello from JNI!");
}

Android.mk

LOCAL_PATH:= $(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE := my_simple_library
LOCAL_ARM_MODE := arm
LOCAL_SRC_FILES := ../../../native/simple_library/simple_library.a
include $(PREBUILT_STATIC_LIBRARY)
#include $(BUILD_STATIC_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE_TAGS    := eng
LOCAL_ARM_MODE       := arm
#LOCAL_PRELINK_MODULE := false
LOCAL_MODULE         := hello-jni
LOCAL_SRC_FILES      := hello-jni.c
LOCAL_C_INCLUDES     := ../../../android/native/simple_library
LOCAL_STATIC_LIBRARIES := my_simple_library
#LOCAL_WHOLE_STATIC_LIBRARIES := my_simple_library
include $(BUILD_SHARED_LIBRARY)

Application.mk

APP_MODULES := my_simple_library hello-jni

就像我说的那样,问题出在我真正利用Java应用程序中jni本机代码中库中的功能时。

现在,我不再确定我的问题是在静态库的makefile中还是在Android.mk中。 我首先认为它一定与库本身的生成有关,但是在这一点上,看到Android.mk中有很多我不知道的选项之后,我不得不承认我毫无头绪。

还有什么..?

哦,是的,我还注意到我的纯C ++库正在使用cpp扩展名,而Java项目中的jni代码正在使用c扩展名。 我试图让后者也使用cpp,但是编译器抱怨。 这可能是问题的一部分吗?

我尝试编译Android.mk文件时收到的错误代码是“未定义符号”,因此hello-jni.c无法看到simple_library.a中的功能列表(实际上有1个功能) C ++而不是普通的C?

您可能会注意到的另一件事是,我正在尝试在自己的Makefile上构建静态库,而不是使用Android.mk生成它。 这是有原因的,但是在这一点上,如果需要这样做的话,我也很高兴如果必须将其由Android.mk生成。

我看不到在此处添加附件以便与项目共享zip的任何方法。 让我知道是否丢失了某些内容。否则,代码位于bitbucket的软件仓库中,因此我也可以轻松地与那里有帐户的任何人共享该代码。

谢谢你的回答。

您写了很多正确的东西,但您只缺少其中的一件。

该函数的名称在C ++中被修饰。 在.so文件中,您不会获得“添加”符号,而是会出现“ Add @ 8i”之类的符号。 为避免损坏,请使用

extern "C" int Add(int x, int y)

.cpp文件和.h中的声明。

通常情况下,还会添加

/// Some .h file
#ifdef __cplusplus
extern "C" {
#endif

/// Your usual C-like declarations go here

#ifdef __cplusplus
} // extern "C"
#endif

而且

extern "C"

.cpp文件中的每个导出函数。

暂无
暂无

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

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