在 JNI 中使用错误代码将 jbytearray 转换为 char*
Convert jbytearray to char* with wrong code in JNI
我尝试在 Jni 层将 jbytearray 转换为 char*。但是,它的结果有一些错误的字符,如下所示:
07-23 10:22:43.701 26585-26646/com.zspirytus.androidlua D/ScriptPkgDataFetcher: getPkgData: 0028 at Kotlin
07-23 10:22:43.701 26585-26646/com.zspirytus.androidlua D/LuaEngine: getPkgData: 0028�ض
07-23 10:22:44.221 26585-26646/com.zspirytus.androidlua D/ScriptPkgDataFetcher: getPkgData: 0028 at Kotlin
07-23 10:22:44.221 26585-26646/com.zspirytus.androidlua D/LuaEngine: getPkgData: 0028�ض
07-23 10:22:44.721 26585-26646/com.zspirytus.androidlua D/ScriptPkgDataFetcher: getPkgData: 0028 at Kotlin
07-23 10:22:44.721 26585-26646/com.zspirytus.androidlua D/LuaEngine: getPkgData: 0028�ض
在Kotlin Layer,String类型的结果是"0028"
;在Jni层,转换后的结果是"0028�ض"
。以下是我在代码中所做的:
在 Kotlin 中,函数 return 一个字节数组:
fun getContentByEntryName(entryName: String): ByteArray {
// data is always a String with value "0028"
val data = ZipFileUtils.getFileContentFromZipFile(ZipFile(scriptPkg), entryName)
Log.d("ScriptPkgDataFetcher", "getPkgDatagetPkgData: $data at Kotlin")
return data.toByteArray()
}
在 Jni,我像这样将 jbytearray 转换为 char*:
jbyteArray jba = (jbyteArray) env->CallStaticObjectMethod(clazz, methodId, dataPath);
int len = env->GetArrayLength (jba);
char* buff = new char[len];
env->GetByteArrayRegion (jba, 0, len, reinterpret_cast<jbyte*>(buff));
Log_d(LOG_TAG, "getPkgData: %s", buff);
似乎工作不正确。我也试过这段代码,但还是不行
工作正常...
jbyteArray jba = (jbyteArray) env->CallStaticObjectMethod(clazz, methodId, dataPath);
const char *cStr = (char *) (env)->GetByteArrayElements(jba, NULL);
我的代码有没有错误?请帮我更正它。谢谢!
如果您要使用 new char[]
创建字符串数据,然后在需要 null-termination 的函数中使用该数据,您必须明确添加空终止符。
此外,您可能没有为终止空分配足够的 space,因此您需要分配 len + 1
字节。
你可以这样做:
char* buff = new char[len + 1]();
用空字符填充缓冲区或:
char* buff = new char[len + 1];
buff[len] = '[=11=]';
不过,我强烈建议您使用 std::string
或 std::vector<char>
而不是这样做。
由于 JNI 非常脆弱,如果抛出异常,使用这种方式使用原始指针可能会导致内存泄漏。
这是一个小例程,可用于将 Java 字节数组转换为向量:
#include <vector>
//...
jbyteArray jba = (jbyteArray) env->CallStaticObjectMethod(clazz, methodId, dataPath);
int len = env->GetArrayLength (jba);
std::vector<char> buff(len + 1, 0);
env->GetByteArrayRegion (jba, 0, len, reinterpret_cast<jbyte*>(buff.data()));
Log_d(LOG_TAG, "getPkgData: %s", buff.data());
如果抛出异常,这不会遭受内存泄漏,因为当向量出于任何原因超出范围时,std::vector
将自动释放内存。
我尝试在 Jni 层将 jbytearray 转换为 char*。但是,它的结果有一些错误的字符,如下所示:
07-23 10:22:43.701 26585-26646/com.zspirytus.androidlua D/ScriptPkgDataFetcher: getPkgData: 0028 at Kotlin
07-23 10:22:43.701 26585-26646/com.zspirytus.androidlua D/LuaEngine: getPkgData: 0028�ض
07-23 10:22:44.221 26585-26646/com.zspirytus.androidlua D/ScriptPkgDataFetcher: getPkgData: 0028 at Kotlin
07-23 10:22:44.221 26585-26646/com.zspirytus.androidlua D/LuaEngine: getPkgData: 0028�ض
07-23 10:22:44.721 26585-26646/com.zspirytus.androidlua D/ScriptPkgDataFetcher: getPkgData: 0028 at Kotlin
07-23 10:22:44.721 26585-26646/com.zspirytus.androidlua D/LuaEngine: getPkgData: 0028�ض
在Kotlin Layer,String类型的结果是"0028"
;在Jni层,转换后的结果是"0028�ض"
。以下是我在代码中所做的:
在 Kotlin 中,函数 return 一个字节数组:
fun getContentByEntryName(entryName: String): ByteArray {
// data is always a String with value "0028"
val data = ZipFileUtils.getFileContentFromZipFile(ZipFile(scriptPkg), entryName)
Log.d("ScriptPkgDataFetcher", "getPkgDatagetPkgData: $data at Kotlin")
return data.toByteArray()
}
在 Jni,我像这样将 jbytearray 转换为 char*:
jbyteArray jba = (jbyteArray) env->CallStaticObjectMethod(clazz, methodId, dataPath);
int len = env->GetArrayLength (jba);
char* buff = new char[len];
env->GetByteArrayRegion (jba, 0, len, reinterpret_cast<jbyte*>(buff));
Log_d(LOG_TAG, "getPkgData: %s", buff);
似乎工作不正确。我也试过这段代码,但还是不行 工作正常...
jbyteArray jba = (jbyteArray) env->CallStaticObjectMethod(clazz, methodId, dataPath);
const char *cStr = (char *) (env)->GetByteArrayElements(jba, NULL);
我的代码有没有错误?请帮我更正它。谢谢!
如果您要使用 new char[]
创建字符串数据,然后在需要 null-termination 的函数中使用该数据,您必须明确添加空终止符。
此外,您可能没有为终止空分配足够的 space,因此您需要分配 len + 1
字节。
你可以这样做:
char* buff = new char[len + 1]();
用空字符填充缓冲区或:
char* buff = new char[len + 1];
buff[len] = '[=11=]';
不过,我强烈建议您使用 std::string
或 std::vector<char>
而不是这样做。
由于 JNI 非常脆弱,如果抛出异常,使用这种方式使用原始指针可能会导致内存泄漏。
这是一个小例程,可用于将 Java 字节数组转换为向量:
#include <vector>
//...
jbyteArray jba = (jbyteArray) env->CallStaticObjectMethod(clazz, methodId, dataPath);
int len = env->GetArrayLength (jba);
std::vector<char> buff(len + 1, 0);
env->GetByteArrayRegion (jba, 0, len, reinterpret_cast<jbyte*>(buff.data()));
Log_d(LOG_TAG, "getPkgData: %s", buff.data());
如果抛出异常,这不会遭受内存泄漏,因为当向量出于任何原因超出范围时,std::vector
将自动释放内存。