0

在本机活动的情况下,我想从 java 调用本机代码。

假设我在engine.so 中有游戏引擎。现在我想添加语音识别。我添加了 java wrapper-class 并通过 jni 从本机代码开始语音识别。我想将结果返回到本机端。在 jni 示例之后,我在 java 类中声明了本机方法,并在识别完成时调用它:

public native void onSpeechRecognized ( String value );

我在engine.so中实现了这个方法。当然,我不会使用 System.loadLibrary 加载 engine.so,因为它已经加载了。但是java代码没有看到方法实现,报:

FATAL EXCEPTION: main
java.lang.UnsatisfiedLinkError: onSpeechRecognized
    at com.company.appname.SpeechRecognizerWrapper.onSpeechRecognized(Native Method)
    at com.company.appname.SpeechRecognizerWrapper$SpeechRecognitionListener.onResults(SpeechRecognizerWrapper.java:92)
    at android.speech.SpeechRecognizer$InternalListener$1.handleMessage(SpeechRecognizer.java:428)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:130)
    at android.app.ActivityThread.main(ActivityThread.java:3687)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:507)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:842)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:600)
    at dalvik.system.NativeStart.main(Native Method)

nm 实用程序显示 engine.so 包含Java_com_company_appname_SpeechRecognizerWrapper_onSpeechRecognized
使用 javah 生成的签名。
我的安卓.mk

PROJ_PATH := $(call my-dir)
LIB_PATH := $(PROJ_PATH)/../../../../../Smart/Lib

include $(LIB_PATH)/Log/Projects/android/jni/Android-prebuilt.mk

...

LOCAL_PATH := $(PROJ_PATH)/../../../../../Smart/Smart
include $(CLEAR_VARS)
LOCAL_C_INCLUDES := \
    $(PROJ_PATH)/../../../../../Smart \
    $(LIB_PATH)/Hash
LOCAL_MODULE    := smart
LOCAL_SRC_FILES := Animation/TextureAnimation.cpp
LOCAL_SRC_FILES += Base/Director.cpp

...

LOCAL_CFLAGS += -DNDEBUG -O3 -mcpu=cortex-a8 -mfpu=neon -ftree-vectorize -mvectorize-with-neon-quad -std=gnu++11
LOCAL_LDLIBS := -llog -landroid -lGLESv2 -lEGL -lOpenSLES
LOCAL_STATIC_LIBRARIES := android_native_app_glue Slb freetype Image FileSystem Noise Log Math Threads SharedPtr vmath png jpeg ScriptEngine QuestEngine Time tremolo

include $(BUILD_SHARED_LIBRARY)

$(call import-module,android/native_app_glue)

那么为什么java端看不到本机端实现呢?

4

2 回答 2

0

我注意到您正在编译 .ccp 文件而不是普通 C。为了让 jni 能够根据函数名称自动查找函数(而不是使用 RegisterNatives 注册),目标文件中的函数名称必须遵循特定的模式。

您是否有可能没有声明函数“extern C”以防止 C++ 名称修改(这是 C++ 对目标文件中的函数重新命名以便它们包含类型信息的做法,这将导致 jni 无法找到函数,因为目标文件中的函数名不再符合正确的模式)。

如果是这样,您可以通过将要从 JNI 调用的函数包围起来来解决问题:

extern "C" { <your function here> }

参见,例如,这个 SO 答案:

Android 应用程序 (ndk) 中本机 cpp 函数的 UnsatisfiedLinkError

您上面 nm 实用程序的输出似乎没有显示为该功能打印的完整行 nm。如果原始 nm 输出中函数名称的任何一侧都有乱码,则可能表明该函数不是“extern C”,并且名称修改是您的麻烦的根源。

于 2015-01-31T12:47:42.623 回答
0

不知道为什么 java wrapper-class 看不到本机函数,但我仅使用RegisterNatives本机代码就解决了这个问题。

于 2015-01-31T10:53:14.450 回答