web-dev-qa-db-ja.com

jstring(JNI)からutf8文字を含むstd :: string(c ++)

std::string文字でjstring(JNI)をutf8(c ++)に変換する方法は?

これは私のコードです。 utf8以外の文字で動作しましたが、utf8の文字では間違っています。

std::string jstring2string(JNIEnv *env, jstring jStr){
    const char *cstr = env->GetStringUTFChars(jStr, NULL);
    std::string str = std::string(cstr);
    env->ReleaseStringUTFChars(jStr, str);
    return str;
}
14
Mr Jerry

解決策を見つけるために多くの時間の後。私は方法を見つけました:

Javaでは、Unicode文字は2バイト(utf16)を使用してエンコードされます。 jstringは、コンテナ文字utf16になります。 c ++のstd::stringは基本的に文字ではなくバイトの文字列であるため、jstringからc++JNIを渡したい場合、utf16をバイトに変換します。

ドキュメント JNI関数 では、jstringから文字列を取得する2つの関数があります。

// Returns a pointer to the array of Unicode characters of the string. 
// This pointer is valid until ReleaseStringchars() is called.
const jchar * GetStringChars(JNIEnv *env, jstring string, jboolean *isCopy);


// Returns a pointer to an array of bytes representing the string 
// in modified UTF-8 encoding. This array is valid until it is released 
// by ReleaseStringUTFChars().
const char * GetStringUTFChars(JNIEnv *env, jstring string, jboolean *isCopy);

GetStringUTFChars修正utf8 を返します。

GetStringCharsはjbyte *を返します。jbytesからcharコードを読み取り、c ++でcharに変換します

これは私の解決策です(asciiおよびutf8文字でうまく機能しました):

std::string jstring2string(JNIEnv *env, jstring jStr) {
    if (!jStr)
        return "";

    const jclass stringClass = env->GetObjectClass(jStr);
    const jmethodID getBytes = env->GetMethodID(stringClass, "getBytes", "(Ljava/lang/String;)[B");
    const jbyteArray stringJbytes = (jbyteArray) env->CallObjectMethod(jStr, getBytes, env->NewStringUTF("UTF-8"));

    size_t length = (size_t) env->GetArrayLength(stringJbytes);
    jbyte* pBytes = env->GetByteArrayElements(stringJbytes, NULL);

    std::string ret = std::string((char *)pBytes, length);
    env->ReleaseByteArrayElements(stringJbytes, pBytes, JNI_ABORT);

    env->DeleteLocalRef(stringJbytes);
    env->DeleteLocalRef(stringClass);
    return ret;
}
17
Mr Jerry
jboolean isCopy;
const char *convertedValue = (env)->GetStringUTFChars(yourJStringParam, &isCopy);
std::string string = std::string(convertedValue, length)

これはうまく機能します。試してみる。

3
Nikhil Tambe

説明はこちらです:

http://www.club.cc.cmu.edu/~cmccabe/blog_jni_flaws.htmlhttps://developer.Android.com/training/articles/perf-jni =(地域の呼び出し)

jsize
string_j2c(JNIEnv *env, jstring p, char *buffer) {

    if (buffer != NULL) {
        // Returns the length (the count of Unicode characters) of a
        // Java string.
        const jsize len = (*env).GetStringLength(p);

        // Translates 'len' number of Unicode characters into modified
        // UTF-8 encoding and place the result in the given buffer.
        (*env).GetStringUTFRegion(p, 0, len, buffer);

        // Returns JNI_TRUE when there is a pending exception;
        // otherwise, returns JNI_FALSE.
        const jboolean is_error = (*env).ExceptionCheck();

        if (is_error == JNI_TRUE) {
            return -1;
        }
    }

    // Returns the length in bytes of the modified UTF-8
    // representation of a string.
    const jsize len = (*env).GetStringUTFLength(p);
    return len;
}

const jsize len = string_j2c(env, p, NULL);
char buffer[len];
const jsize ret = string_j2c(env, p, buffer);

if (ret == -1) { // error
}
else {
    __Android_log_print(Android_LOG_DEBUG, "Native", "%s", buffer);
}