JNI之数组与字符串的使用

字符串和数组是JNI中常见的引用数据类型,本文将介绍符串和数组在JNI中的常见处理方式。

JNI中字符串的处理

1、Java字符串与原生字符串转换

当从java层传递一个字符串过来之后,它的类型是jstring,同样如果需要返回一个字符串给java层,它的类型也是jstring。jstring代表着Java虚拟机中的一个字符串,并且不同于C++语言的string类型。

如果原生代码需要处理jstring,需要通过JNIEnv将其转换为原生字符串才可以使用。通过JNI函数GetStringUTFChars来读取这个字符串中的内容,GetStringUTFChars函数可以通过JNIEnv接口指针调用,它将一个代表着Java虚拟机中的字符串jstring引用,转换成为一个UTF-8形式的C字符串。

当原生代码使用完了通过GetStringUTFChars获取的原生字符串后应该使用ReleaseStringUTFChars释放它。调用ReleaseStringUTFChars标识着原生代码不再需要使用从GetStringUTFChars获取的UTF-8字符串了,这个UTF-8字符串所占用的空间就可以被释放了。
如果不调用ReleaseStringUTFChars释放原生字符串的话将会导致内存泄露。

我们看下函数GetStringUTFChars的原型是:

const char* GetStringUTFChars(jstring string, jboolean* isCopy)

在这里第三个参数表示如果返回的字符串是原来的java.lang.String的一份拷贝,则在函数GetStringUTFChars返回之后,isCopy指向的内存地址将会被设置为JNI_TRUE。而如果返回的字符串指针直接指向原来的java.lang.String对象,则该地址会被设置为JNI_FALSE.如果返回了JNI_FALSE, 则原生代码将不能改变返回的字符串,因为改变了这个字符串,原来的java字符串也会被修改,这违背了java.lang.String实例不可改变的原则。
通常你可以直接传递NULL给isCopy来告诉Java虚拟机你不在乎返回的字符串是否指向原来Java的String对象。

如果需要将C/C++的字符串返回给Java层,则需要通过函数NewStringUTF生成jstring返回。

例如下面的例子展示了在Native层获取java层字符串,并修改返回给java层的一个例子:

public class MainActivity extends AppCompatActivity {

    static {
        System.loadLibrary("jnitest");
    }
    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = ActivityMainBinding.inflate(getLayoutInflater());
        setContentView(binding.getRoot());
        TextView tv = binding.sampleText;
        try {
            // 尽量确保传递进去的是utf-8字符串
            tv.setText(sayHello(new String("James".getBytes(),"utf-8")));
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
    }
    public native String sayHello(String name);
}
extern "C"
JNIEXPORT jstring JNICALL
Java_com_fly_jnitest_MainActivity_sayHello(JNIEnv *env, jobject thiz, jstring name) {
    // 将java字符串转换成Native层字符串
    std::string str = "hello ";
    const char *cname = env->GetStringUTFChars(name, nullptr);
    str.append(cname);
    // 释放字符串
    env->ReleaseStringUTFChars(name,cname);
    return env->NewStringUTF(str.c_str());
}

除了上面介绍的GetStringUTFChars, ReleaseStringUTFChars以及NewStringUTF, JNI还提供了
GetStringCharsReleaseStringChars等相关API处理Unicode格式的字符串。

下表是JNI中常用的一些操作字符串的相关API:

JNI函数描述
Get/ReleaseStringChars获取或者释放一个Unicode格式的字符串,可能返回原始字符串的拷贝
Get/ReleaseStringUTFChars获取或者释放一个UTF-8格式的字符串,可能返回原始字符串的拷贝
GetStringLength返回Unicode字符串的字符个数
GetStringUTFLength返回用于表示某个UTF-8字符串所需要的字节个数(不包括结束的0)
NewString创建一个java.lang.String对象,该对象与指定的Unicode字符串具有相同的字符序列
NewStringUTF创建一个java.lang.String对象,该对象与指定的UTF-8字符串具有相同的字符序列
Get/ReleaseStringCritical获取或者释放一个Unicode格式的字符串的内容,可能返回原始字符串的拷贝,在Get/ReleaseStringCritical之间的代码必须不能阻塞
Get/SetStringRegion将一个字符串拷贝到预先开辟的空间,或者从一个预先开辟的空间复制字符串,字符使用Unicode编码
Get/SetStringUTFRegion将一个字符串拷贝到预先开辟的空间,或者从一个预先开辟的空间复制字符串,字符使用UTF-8编码

JNI中数组的处理

在JNI中使用jarray以及像jintArray等子类表示数组。正如jstring不是一个C/C++的字符串类型,jarray也不是C/C++的数组类型。如果需要在native处理数组,同样需要通过JNIEnv接口将jarray转换。

例如下面是一个展示了计算一个java数组之和的例子:

extern "C"
JNIEXPORT jint JNICALL
Java_com_fly_jnitest_MainActivity_sum(JNIEnv *env, jobject thiz, jintArray array) {
    jint length = env->GetArrayLength(array);
    jint c_array[length];
    env->GetIntArrayRegion(array,0,length,c_array);
    int sum = 0;
    for (int i = 0; i < length; ++i) {
        sum+= c_array[i];
    }
    return sum;
}

对于上述累计数组和的例子,使用函数GetIntArrayElements获取数组元素实现也是可以的,但是需要注意的是GetIntArrayElements要和ReleaseIntArrayElements配对使用,以免造成内存泄漏。

以下这个例子展示了在JNI函数中排序数组,然后将排序好的数组同步到java层的功能:

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_MainActivity_changeArray(JNIEnv *env, jobject thiz, jintArray array) {
    // 方法一
//    jint length = env->GetArrayLength(array);
//    jint c_array[length];
//    env->GetIntArrayRegion(array,0,length,c_array);
//    std::sort(c_array,c_array + length);
//    env->SetIntArrayRegion(array,0,length,c_array); // 数组同步,不然java层的数组不会改变

    // 方法二
    jint length = env->GetArrayLength(array);
    jint *c_array = env->GetIntArrayElements(array, nullptr);
    std::sort(c_array,c_array + length);
    env->SetIntArrayRegion(array,0,length,c_array); // 同步
    env->ReleaseIntArrayElements(array,c_array,0); // 释放
}

在上面的例子中GetIntArrayRegionGetIntArrayElements都可以获取到数组相关元素,那么他们有什么区别呢?

对于小量的、固定大小的数组,应该选择Get/SetArrayRegion系列函数来操作数组元素是效率最高的。因为这对函数要求提前分配一个C/C++临时缓冲区来存储数组元素,开发者可以直接在栈上或在堆上来动态申请,当然在栈上申请是最快的。有童鞋可能会认为,访问数组元素还需要将原始数据全部拷贝一份到临时缓冲区才能访问而觉得效率低?其实这种复制少量数组元素的代价是很小的,几乎可以忽略。这对函数的另外一个优点就是,允许你传入一个开始索引和长度来实现对子数组元素的访问和操作(SetArrayRegion函数可以修改数组),不过传入的索引和长度不要越界,函数会进行检查,如果越界了会抛出 ArrayIndexOutOfBoundsException 异常。
Get/ReleaseArrayElements系列函数永远是安全的,JVM会选择性的返回一个指针,这个指针可能指向原始数据,也可能指向原始数据的复制,更加适用于数据量比较大的数组。

下表是JNI中常用的一些操作数组的相关API:

JNI函数功能描述
Get/SetArrayRegion复制基础类型数组的内容到C缓冲区或者将C缓冲区的内容设置到基础类型数组中去
Get/ReleaseArrayElements获取/释放指向基础类型数组内容的指针,可能返回原始数组内容的拷贝
GetArrayLength返回数组中元素的个数
NewArray创建指定长度的数组

JNI为访问对象数组单独提供了一组单独的函数GetObjectArrayElement返回指定下表的元素, 而SetObjectArrayElement则修改指定索引上的元素。对于引用类型数组与基础类型数组不同的是,你不能一次获取或者拷贝对象数组中的所有元素,需要使用Get/SetObjectArrayElement来访问引用类型的数组。

推荐阅读

JNI基础简介

关注我,一起进步,人生不止coding!!!
微信扫码关注

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JNI 中,可以使用 `jstring` 类型来表示 Java 字符串。如果你想将这个字符串转换为 UTF-16 字符数组,可以使用 `GetStringChars` 函数来获取 `jchar` 类型的指针,然后将其转换为 UTF-16 字符数组。 具体来说,你可以按照以下步骤实现: 1. 获取 `jstring` 对象:在 JNI 中,可以通过调用 `JNIEnv` 的 `NewStringUTF` 或 `NewString` 函数创建一个 Java 字符串对象,并将其转换为 `jstring` 类型。 2. 获取 UTF-16 字符数组指针:使用 `GetStringChars` 函数来获取 `jchar` 类型的指针,该指针指向字符串UTF-16 字符数组。 3. 转换为 UTF-16 字符数组:根据需要,你可以将 `jchar` 指针转换为 `wchar_t` 或其他类型的字符指针,以便在 C/C++ 代码中使用。 下面是一个示例代码,演示如何在 JNI 中获取字符串UTF-16 字符数组指针: ```c++ jstring jstr = env->NewStringUTF("Hello, world!"); // 创建 jstring 对象 const jchar* jchars = env->GetStringChars(jstr, NULL); // 获取 UTF-16 字符数组指针 wchar_t* wchars = new wchar_t[env->GetStringLength(jstr) + 1]; // 分配内存 for (int i = 0; i < env->GetStringLength(jstr); i++) { wchars[i] = static_cast<wchar_t>(jchars[i]); // 转换为 wchar_t 类型 } wchars[env->GetStringLength(jstr)] = L'\0'; // 添加字符串结尾符号 env->ReleaseStringChars(jstr, jchars); // 释放 jchar 指针 ``` 需要注意的是,使用完 `jchar` 指针后,一定要调用 `ReleaseStringChars` 函数来释放资源。此外,在转换为其他类型的字符指针时,需要根据具体的编码方式进行转换。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值