Android ndk 字符串传参-阿里云开发者社区

开发者社区> 开发与运维> 正文

Android ndk 字符串传参

简介: ndk开发的过程中,Android(Java)与JNI(C/C++)之间,由于参数类型无法互通,所以在传参的过程中,会遇到不少的麻烦。

ndk开发的过程中,Android(Java)与JNI(C/C++)之间,由于参数类型无法互通,所以在传参的过程中,会遇到不少的麻烦。


其中我感觉最麻烦的要数字符串类型的传参



从JAVA传字符串到C++

上层JAVA数据类型String对应NDK为jstring类型,上层传参的方式和普通的java函数间调用并没有什么区别,重点是底层函数,如何将上层传来的jstring转化成本地可认并且可以操作的char *数组


例子如下(函数封装,可由实际函数调用。返回的char *数组一定要在外部记得释放,否则会造成内存泄漏):

char *JavaStringToCStr(JNIEnv *env, jstring str) {
 
    char *result = NULL;
 
    //  先找到JAVA/String类
    jclass strCls = env->FindClass("java/lang/String");
    jstring encodeMothed = env->NewStringUTF("UTF8");
 
    jmethodID java_getBytes = env->GetMethodID(strCls, "getBytes", "(Ljava/lang/String;)[B");
    jbyteArray barr = (jbyteArray)env->CallObjectMethod(str, java_getBytes, encodeMothed);
    jsize length = env->GetArrayLength(barr);
 
    jbyte *ba = env->GetByteArrayElements(barr, JNI_FALSE);
    if (length > 0) {
        result = (char *)malloc(length + 1);
        memcpy(result, ba, length);
        result[length] = 0;
 
        env->ReleaseByteArrayElements(barr, ba, 0);
    }
 
    return result;
 
}


主题思想:先将String转化成JAVA的字符串数组(对应java描述为byte[]),再对byte进行字符串复制操作。

方法:

    1、定位JAVA的String类

    2、找到String类下面的函数getBytes

    3、调用getBytes将字符串转化成jbyte

    4、进行字符串复制操作

    5、释放相关资源

从C++传字符串到java


        从C++传字符串到java又分为两种形式

        1、通过result返回值的方式

            这种方式其实在使用的时候比较简单,如下即可:

return env->NewStringUTF("jni:string from jni");

        2、通过回调的方式向JAVA传字符串

        这种传值的方式就比较麻烦。回调的方法此处不谈,仅聊参数的传递

        实例如下:

void Java_com_example_ndktest_JniCommon_toJniString(JNIEnv *env, jobject thiz, jstring str) {
 
    char*cStr = JavaStringToCStr(env, str);
 
    constchar *demo = "jni say:";
 
    char*result = (char*)malloc(strlen(demo) +strlen(cStr) + 1);
    memset(result, 0,strlen(demo)+strlen(cStr)+1);
    strcpy(result, demo);
    if(cStr != NULL) {
        strcat(result, cStr);
    }
 
    jstring java_result = env->NewStringUTF(result);
 
    //  调用上层java
    jclass clazz = env->GetObjectClass(thiz);
    jclass cClazz = (jclass)env->NewGlobalRef(clazz);
 
    jmethodID ShowJniString = env->GetMethodID(cClazz,"ShowJniString", "(Ljava/lang/String;)V");
 
    env->CallVoidMethod(thiz, ShowJniString, java_result);
 
    free(cStr);
    free(result);
 
}


重点:其实回调的方式和普通函数的回调类似,重点在于env->GetMethodID这个函数的第三个参数写法。其中括号内代表参数,多个可以直接连续写下去。括号外代表返回值

( arg-types ) ret-type


开发过程中遇到的JAVA类型标识列表

标识 类型 举例
J long -----
Z boolean ----
B byte -----
C char -----
S short -------
I int -----
F float ----
D double ------
Lclassname; specify class Ljava/lang/String;
[type type[] 本文第一段代码:[B
     

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

分享:
开发与运维
使用钉钉扫一扫加入圈子
+ 订阅

集结各类场景实战经验,助你开发运维畅行无忧

其他文章