zoukankan      html  css  js  c++  java
  • 网摘关于JNI理解

    JAVA基础之理解JNI原理 http://www.cnblogs.com/mandroid/archive/2011/06/15/2081093.html


    JNI技术实践小结  http://www.iteye.com/topic/304594#814127


    Android JNI知识简介  http://www.cnblogs.com/hoys/archive/2010/10/28/1863612.html


    android JNI编程 一些技巧(整理) http://blog.csdn.net/linweig/article/details/5203716

      一:java中String 到 C中的 char *

      java中方法:String getLine(String prompt);

      javah -jni 得到头文件中的声明是:JNIEXPORT jstring JNICALL Java_Prompt_getLine(JNIEnv *env, jobject this, jstring prompt);

      jstring是JNI中对应于java中的String类型,与C++的string不同。

      

    JNIEXPORT jstring JNICALL Java_Prompt_getLine(JNIEnv *env, jobject obj, jstring prompt)
    
    {
    
        const char* str;
    
        str = env->GetStringUTFChars(prompt, false);
    
        if(str == NULL) {
    
            return NULL; /* OutOfMemoryError already thrown */
    
        }
    
        std::cout << str << std::endl;
    
        env->ReleaseStringUTFChars(prompt, str);
    
     
    
        char* tmpstr = "return string succeeded";
    
        jstring rtstr = env->NewStringUTF(tmpstr);
    
        return rtstr;
    
    }
    

      在上面的例子中,作为参数的prompt不能直接被C++程序使用,先做了如下转换
      str = env->GetStringUTFChars(prompt, false);
      将jstring类型变成一个char*类型。

      返回的时候,要生成一个jstring类型的对象,也必须通过如下命令,
      jstring rtstr = env->NewStringUTF(tmpstr);

      GetStringUTFChars和NewStringUTF都是JNI提供的处理String类型的函数,还有其他的函数

      二:数组的处理

      

    和String一样,JNI为Java基本类型的数组提供了j*Array类型,比如int[]对应的就是jintArray。来看一个传递int数组的例子,Java程序就不写了,
    
    JNIEXPORT jint JNICALL Java_IntArray_sumArray(JNIEnv *env, jobject obj, jintArray arr)
    
    {
    
        jint *carr;
    
        carr = env->GetIntArrayElements(arr, false);
    
        if(carr == NULL) {
    
            return 0; /* exception occurred */
    
        }
    
        jint sum = 0;
    
        for(int i=0; i<10; i++) {
    
            sum += carr[i];
    
        }
    
        env->ReleaseIntArrayElements(arr, carr, 0);
    
        return sum;
    
    }
    
    这个例子中的GetIntArrayElements和ReleaseIntArrayElements函数就是JNI提供用于处理int数组的函 数。如果试图用arr[i]的方式去访问jintArray类型,毫无疑问会出错。
    JNI还提供了另一对函数GetIntArrayRegion和 ReleaseIntArrayRegion访问int数组,就不介绍了,对于其他基本类型的数组,方法类似。
    

      三:二维数组和String数组

      

    在JNI中,二维数组和String数组都被视为object数组,因为数组和String被视为object。仍然用一个例子来说明,这次是一个二维int数组,作为返回值。
    
    JNIEXPORT jobjectArray JNICALL Java_ObjectArrayTest_initInt2DArray(JNIEnv *env, jclass cls, int size)
    
    {
    
        jobjectArray result;
    
        jclass intArrCls = env->FindClass("[I");
    
        result = env->NewObjectArray(size, intArrCls, NULL);
    
     
    
        for (int i = 0; i < size; i++) {
    
            jint tmp[256]; /* make sure it is large enough! */
    
            jintArray iarr = env->NewIntArray(size);
    
            for(int j = 0; j < size; j++) {
    
                tmp[j] = i + j;
    
            }
    
            env->SetIntArrayRegion(iarr, 0, size, tmp);
    
            env->SetObjectArrayElement(result, i, iarr);
    
            env->DeleteLocalRef(iarr);
    
        }
    
        return result;
    
    }
    
    上面代码中的第三行,
    jobjectArray result;
    因为要返回值,所以需要新建一个jobjectArray对象。
    
    jclass intArrCls = env->FindClass("[I");
    是创建一个jclass的引用,因为result的元素是一维int数组的引用,所以intArrCls必须是一维int数组的引用,这一点是如何保证的 呢?注意FindClass的参数"[I",JNI就是通过它来确定引用的类型的,I表示是int类型,[标识是数组。对于其他的类型,都有相应的表示方 法,
    
    Z boolean 
    B byte 
    C char 
    S short 
    I int 
    J long 
    F float 
    D double
    
    String是通过“Ljava/lang/String;”表示的,那相应的,String数组就应该是“[Ljava/lang/String;”。
    
    还是回到代码,
    result = env->NewObjectArray(size, intArrCls, NULL);
    的作用是为result分配空间。
    
    jintArray iarr = env->NewIntArray(size);
    是为一维int数组iarr分配空间。
    
    env->SetIntArrayRegion(iarr, 0, size, tmp);
    是为iarr赋值。
    
    env->SetObjectArrayElement(result, i, iarr);
    是为result的第i个元素赋值。
    
    通过上面这些步骤,我们就创建了一个二维int数组,并赋值完毕,这样就可以做为参数返回了。
    
    如果了解了上面介绍的这些内容,基本上大部分的任务都可以对付了。虽然在操作数组类型,尤其是二维数组和String数组的时候,比起在单独的语言中编程要麻烦,但既然我们享受了跨语言编程的好处,必然要付出一定的代价。
    
    有一点要补充的是,本文所用到的函数调用方式都是针对C++的,如果要在C中使用,所有的env->都要被替换成(*env)->,而 且后面的函数中需要增加一个参数env,具体请看一下jni.h的代码。另外还有些省略的内容,可以参考JNI的文档:Java Native Interface 6.0 Specification,在JDK的文档里就可以找到。
    

      

      

    ezhong的博客园:http://www.cnblogs.com/ezhong/


  • 相关阅读:
    Visual Stdio VS 错误 error : 0xC00000FD: Stack overflow. 更改堆栈空间解决栈溢出问题
    OpenCV Mat 只能用静态数组定义时初始化,动态数组赋值给Mat需要逐元素进行. MATLAB,OpenCV,VS混合编程
    【转】Ubuntu 10.10升级显卡驱动后开机动画低分辨率问题
    linux 文件[名]编码
    L337 Speak及国外论坛、IRC常用缩写
    UCS2 手机SMS的PDU编码
    setuid
    【转】CentOS5.5硬盘安装
    SWT CTabFolder 简记
    [转] 程序员的十层楼
  • 原文地址:https://www.cnblogs.com/ezhong/p/2259403.html
Copyright © 2011-2022 走看看