今天,简单讲讲Android jni如何将java的Sring转成char*。
这个之前一直不理解,所以我都是android传入byte[]转成char*,有一篇博客专门讲了这个。后来发现用String转成char*更加简单。这里记录一下。
C++版本如下:
//将char类型转换成jstring类型
jstring CStr2Jstring( JNIEnv* env, const char* pat )
{
// 定义java String类 strClass
jclass strClass = (env)->FindClass("Ljava/lang/String;");
// 获取java String类方法String(byte[],String)的构造器,用于将本地byte[]数组转换为一个新String
jmethodID ctorID = (env)->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V");
// 建立byte数组
jbyteArray bytes = (env)->NewByteArray((jsize)strlen(pat));
// 将char* 转换为byte数组
(env)->SetByteArrayRegion(bytes, 0, (jsize)strlen(pat), (jbyte*)pat);
//设置String, 保存语言类型,用于byte数组转换至String时的参数
jstring encoding = (env)->NewStringUTF("GB2312");
//将byte数组转换为java String,并输出
return (jstring)(env)->NewObject(strClass, ctorID, bytes, encoding);
}
char * Jstring2CStr( JNIEnv * env, jstring jstr )
{
char * rtn = NULL;
jclass clsstring = env->FindClass("java/lang/String");
jstring strencode = env->NewStringUTF("GB2312");
jmethodID mid = env->GetMethodID(clsstring, "getBytes", "(Ljava/lang/String;)[B");
jbyteArray barr= (jbyteArray)env->CallObjectMethod(jstr,mid,strencode);
jsize alen = env->GetArrayLength(barr);
jbyte * ba = env->GetByteArrayElements(barr,JNI_FALSE);
if(alen > 0)
{
rtn = (char*)malloc(alen+1); //new char[alen+1];
memcpy(rtn,ba,alen);
rtn[alen]=0;
}
env->ReleaseByteArrayElements(barr,ba,0);
return rtn;
}
这个相当于将string转成byte,然后再转成char*,应该可以处理中文字符。如果只有英文字符,其实更加简单。
使用GetStringUTFChars将jstring转换成为UTF-8格式的char*,这样就可以直接使用了。用完后需要释放资源,使用ReleaseStringUTFChars释放指向UTF-8格式的char*的指针。不过这个不可以处理中文字符。
具体讲解GetStringUTFChars这个函数:
const char* GetStringUTFChars(JNIEnv*env, jstring string, jboolean *isCopy);
对第三个参数 jboolean *isCopy说明如下:
当从JNI函数GetStringUTFChars函数中返回得到字符串B时,如果B是原始字符串java.lang.String的一份拷贝,
则isCopy 被赋值为JNI_TRUE。如果B是和原始字符串指向的是JVM中的同一份数据,则isCopy 被赋值为JNI_FALSE。
当isCopy 为JNI_FALSE时,本地代码绝不能修改字符串的内容,否则JVM中的原始字符串也会被修改,这会打破Java语言
中字符串不可变的规则。
通常,我们不必关心JVM是否会返回原始字符串的拷贝,只需要为isCopy传递NULL或者0作为参数 。
android jni jstring 转 char*就讲完了。
就这么简单。