【问题标题】:How to call Java function from c如何从c调用Java函数
【发布时间】:2012-02-16 01:19:57
【问题描述】:

我被这个困住了,我需要从 c/c++ 调用一个 Java 函数。

在示例和教程中,我只看到一个调用 c 方法的 java 应用程序,并且在同一个方法中调用另一个 java 方法,但我想做的是从代码的任何部分调用 java 方法。这就是我所拥有的:

static JNIEnv mEnv;
static jclass mClassAndroidActivity;
static mMethodSayHello;
JNIEXPORT void JNICALL JNI_FUNCTION(AndroidActivity_nativeInit)(JNIEnv* env, jobject obj, int width, int height)
{
    mEnv = env;
    jclass cls = (*env)->GetObjectClass(env, obj);
    mClassAndroidActivity = (*env)->NewGlobalRef(env, cls);
    mMethodSayHello = (*env)->GetMethodID (env, mClassAndroidActivity, "SayHello", "(Ljava/lang/String;)V");
}

//this method is called from a cpp
void nativeSayHello(char* msg)
{
    jstring string = (*mEnv)->NewStringUTF(mEnv, msg);
    (*mEnv)->CallVoidMethod(mEnv, mClassAndroidActivity, mMethodSayHello, string);
}

但总是崩溃,我尝试不使用 NewGlobalRef,在 JNI_Function 中使用 mEnv 而不是 env,我尝试从 JNI_OnLoad 获取方法 ID,但总是崩溃。

这是我得到的日志:

02-15 18:09:48.520: W/dalvikvm(27904): JNI WARNING: threadid=1 using env from threadid=0

【问题讨论】:

  • '我得到的“最佳”日志说线程 1 的 env 与线程 0 不同'这是正确的,我相信。每个 Java 线程应该有一个不同的 env 指针。
  • 是的,但是我怎样才能保存第一个 env 指针,以便我可以在不同的线程中使用

标签: java android c++ java-native-interface


【解决方案1】:

您不能重复使用JNIEnv,因为它特定于调用线程。要从本机代码调用(非静态)Java 方法,您需要这样的东西:

static JavaVM *gJavaVM;
static jobject gCallbackObject = NULL;

JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    gJavaVM = vm;
    return JNI_VERSION_1_6;
}

JNIEXPORT void JNICALL JNI_FUNCTION(AndroidActivity_nativeInit)(JNIEnv* env, jobject obj, int width, int height) {
    // ...
    gCallbackObject = (*env)->NewGlobalRef(env, obj);
}

JNIEXPORT void JNICALL JNI_FUNCTION(AndroidActivity_nativeRelease)(JNIEnv* env, jobject obj) {
    (*env)->DeleteGlobalRef(env, gCallbackObject);
    gCallbackObject = NULL;
}

//this method is called from native code
void nativeSayHello(char* msg) {
    int status;
    JNIEnv *env;
    int isAttached = 0;

    if (!gCallbackObject) return;

    if ((status = (*gJavaVM)->GetEnv(gJavaVM, (void**)&env, JNI_VERSION_1_6)) < 0) {
        if ((status = (*gJavaVM)->AttachCurrentThread(gJavaVM, &env, NULL)) < 0) {
            return;
        }
        isAttached = 1;
    }

    jclass cls = (*env)->GetObjectClass(env, gCallbackObject);
    if (!cls) {
        if (isAttached) (*gJavaVM)->DetachCurrentThread(gJavaVM);
        return;
    }

    jmethodID method = (*env)->GetMethodID(env, cls, "SayHello", "(Ljava/lang/String;)V");
    if (!method) {
        if (isAttached) (*gJavaVM)->DetachCurrentThread(gJavaVM);
        return;
    }

    jstring string = (*mEnv)->NewStringUTF(mEnv, msg);
    (*env)->CallVoidMethod(env, gCallbackObject, method, string);

    if (isAttached) (*gJavaVM)->DetachCurrentThread(gJavaVM);
}

此代码 sn-p 未经测试。为防止内存泄漏,当不再需要对该对象的引用时,不要忘记在 Java 代码中调用 nativeRelease() 方法。

有关详细信息,请参阅The Java Native Interface 文档。

【讨论】:

    猜你喜欢
    • 2010-10-23
    • 1970-01-01
    • 2017-04-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-04-25
    • 1970-01-01
    相关资源
    最近更新 更多