Android jni 编程3(对基本类型一维整型数组的操作)总结版

主要学习资料:黑马程序员的NDK方法使用(生产类库so)

                jni编程指南中文版(已上传至博客园)

       博主文章(它使用的是VS和eclipse联合开发):http://www.cnblogs.com/activity-life/p/3643047.html

//0.传入一维整型数组,无返回值(但已对数组进行了修改)
public native void arrayEncode(int[] arr);

//1.传入一维整型数组,数组长度(因为c不容易获取而Java方便),返回整型
private native int SumArray(int [] ar, int length);//传一个数组的长度


//2.传入一维整型数组,无数组长度,返回整型
private native int SumArrayA(int [] ar);//不传长度

在MainActivity.java中声明好本地方法(由C语言实现)

src文件夹下javah指令编译生成.h(方便使用函数名)

配置NDK,(生成类库,jni,libs文件夹)

修改后缀名:.cpp->.c   android.mk 源文件类型修改为.c

C代码:

//0.无返回值,输入一个数组,无长度
JNIEXPORT void JNICALL Java_com_swust_array_MainActivity_arrayEncode
  (JNIEnv * env, jobject obj, jintArray jintarr){
    //拿到整型数组的长度以及第0个元素的地址
         //jsize       (*GetArrayLength)(JNIEnv*, jarray);
        int length = (*env)->GetArrayLength(env, jintarr);
         //jint*       (*GetIntArrayElements)(JNIEnv*, jintArray, jboolean*);
        int* arrp = (*env)->GetIntArrayElements(env, jintarr, 0);

        int i;
        for(i = 0;i < length; i++){
            *(arrp + i) += 10;
        }
}
//1.有返回值int类型,输入一个数组,有长度
JNIEXPORT jint JNICALL Java_com_swust_array_MainActivity_SumArray
  (JNIEnv *env, jobject obj, jintArray arr, jint length ){
     int i, sum = 0;
        jint* pOutbuf = NULL;
        if (length > 0)
            pOutbuf = (jint*) malloc(length * sizeof(jint));
        else
            return 0;
        (*env)->GetIntArrayRegion(env, arr, 0, length, pOutbuf);
        for (i = 0; i < length; i++)
            sum += pOutbuf[i];
        free(pOutbuf);
        pOutbuf = NULL;

        return sum;
}
//2.有返回值int类型,输入一个数组,无长度
JNIEXPORT jint JNICALL Java_com_swust_array_MainActivity_SumArrayA
  (JNIEnv *env, jobject obj, jintArray arr){

    int i,j, sum = 0;
         jint* buf;
         j = (*env)->GetArrayLength(env,arr);
         //另外,这里返回的值,是jint*,不是const jint*,也就是说可以对其中的值进行修改
         buf = (*env)->GetIntArrayElements(env, arr, NULL);//这一句其实也告java内存回收器,不要回收arr数组的内存,或者不要整理内存
                                                           //如果回收器,回收和整理内存,那么我们在c中,访问的地址就可能错了
         for (i = 0; i < j; i++)//这里是求合
                 sum += buf[i];
         //现在我们来修改一下buf中的元素,看看返回后,会不会把buf中的值,更新到java中的arr里去
         buf[0] = 100;
         buf[1] = 200;

         (*env)->ReleaseIntArrayElements(env, arr, buf, 0);//调用这个方法,告诉java内存回收器,我已用完了arr,,现在可以回收arr的内存了
         return sum;
}

导入include(方便查找定义,无警告错误)

在Java代码中调用本地方法

int[] arr = {1,2,3,4,5};

    public void click1(View v){
        //类型1:返回值为void型,输入一维数组无长度
        arrayEncode(arr);
        for (int i : arr) {
            System.out.println(i);
        }
        //类型2:返回值为int型,输入一维数组有长度
        Integer val = SumArray(arr,5);
        Log.d("wsq", "传入数组长度后,和为 = "+ val);
        
        //类型3:返回值为int型,输入一维数组无长度
        //传入一维数组,累计和,并修改了第0,第1个数组元素的值
        Integer value = SumArrayA(arr);
        Toast.makeText(this, value.toString(), 0).show();
        Log.d("wsq", "arr[0] = "+ arr[0]+" arr[1]"+arr[1]);
    }
    

效果:

调用类型1:

类型2:

类型3:

没有学透,以应用为主,大家多多指教,2015-10-25 15:39手记

原文地址:https://www.cnblogs.com/shuqingstudy/p/4908862.html