【问题标题】:JNI String return valueJNI 字符串返回值
【发布时间】:2026-01-12 13:05:01
【问题描述】:

我有一个返回字符串的 Java 实例方法,我通过 C++ 中的 JNI 调用此方法。我写了以下代码:

const char *DiagLayerContainer_getDESC(JNIEnv *env, jobject diagLayer) {
    jclass diagLayerClass = env->FindClass(PARSER_CLASS);
    jmethodID getDESCDiagLayerMethodID = env->GetMethodID(diagLayerClass, "getDESCDiagLayer", "(Ljava/lang/Object;)Ljava/lang/String;");
    jstring returnString = (jstring) env->CallObjectMethod(diagLayer, getDESCDiagLayerMethodID);
    return env->GetStringUTFChars(returnString, JNI_FALSE);
}

如何获取字符串并将其转换为 const char *?

我的程序在最后一行崩溃,访问冲突到 0x00000000。 returnString 不为 NULL。

【问题讨论】:

  • 在调试器中运行您的代码,并在崩溃点检查进程的状态。
  • 这里崩溃了 (jni.h): const char* GetStringUTFChars(jstring str, jboolean *isCopy) { return functions->GetStringUTFChars(this,str,isCopy); } 没有机会深入了解调试器

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


【解决方案1】:

根据GetStringUTFChars,最后一个参数是指向jboolean的指针。

改变

return env->GetStringUTFChars(returnString, JNI_FALSE);

return env->GetStringUTFChars(returnString, NULL);

或者更好的是,返回 std::string

std::string DiagLayerContainer_getDESC(...) {
    ...
    const char *js = env->GetStringUTFChars(returnString, NULL);
    std::string cs(js);
    env->ReleaseStringUTFChars(returnString, js);
    return cs;
}

我已经构建了一个类似的简单示例,目前的代码看起来还不错。

虽然,有两个可能的错误来源。

第一个是方法签名。试试"()Ljava/lang/String;" 而不是"(Ljava/lang/Object;)Ljava/lang/String;"

第二个是在 java 源代码中。如果 java 方法返回空字符串,CallObjectMethod() 将返回 NULL jstringGetStringUTFChars() 失败。

添加一个

if (returnString == NULL)
    return NULL;

CallObjectMethod()之后。

所以查看java源代码,看看getDESCDiagLayer()方法是否可能返回一个空字符串。

【讨论】:

  • 如果isCopy 为假并且returnString 超出范围(即通过显式或隐式PopLocalFrame),您将有一个悬空指针。你应该strdup()GetStringUTFChars 的结果,返回它,然后调用ReleaseStringUTFChars。稍后,像往常一样在本机代码中使用free() 处理副本。
  • strdup() 被替换为 std::string(),它将复制新分配的字符串本身。
  • 如何在 Logcat 中打印 returnString?
  • Logcat 是什么意思? Android 的 Logcat?
最近更新 更多