美文网首页
JNI基础-缓存策略

JNI基础-缓存策略

作者: 伊泽瑞额 | 来源:发表于2020-02-20 12:30 被阅读0次

1. 数组的细节处理

c中的排序方法:

void qsort(void * _Base, size_t _NumOfElements, size_t _SizeOfElements, int(* _PtFuncCompare)(const void *, const void ))
对jarray 排序()第一个参数 void
数组首地址,
第二个参数: 数组的大小长度
第三个参数:数据元素类型的大小
第四个参数: 数组的一个比较方法指针( Comparable)

c中排序完后要同步到jarray 中并且释放 inArray:

同步数组的数据给java 数组,intArray 并不是jarray 可以理解为copy 了一份
0 :既要同步数据给jarray ,又要释放 inArray
JNI_COMMIT:同步苏剧给jarray,也会释放inArray
JNI_ABORT: 不同步数据给jarray,但是会释放inArray
(*env)->ReleaseIntArrayElements(env, jarray, inArray,0);

int comparable(const jint *number1,const jint *number2 ){

    return *number1 - *number2;
}

JNIEXPORT void  JNICALL Java_com_Amn_ndk_03_Simple_sort
(JNIEnv *env, jclass jclz, jintArray jarray){

    
    //获取传过来的数组首地址
    jint * inArray = (*env)->GetIntArrayElements(env, jarray, NULL);
    //获取传过来的数组长度
    int length=(*env)->GetArrayLength(env,jarray);

    //对jarray 排序()第一个参数 void* 数组首地址,
    //第二个参数 数组的大小长度
    //第三个参数:数据元素类型的大小
    //第四个参数: 数组的一个比较方法指针( Comparable)
    qsort(inArray, length, sizeof(int), comparable);

    //同步数组的数据给java 数组,intArray 并不是jarray 可以理解为copy 了一份
    //0 :既要同步数据给jarray ,又要释放 inArray 
    //JNI_COMMIT:同步苏剧给jarray,也会释放inArray 
    //JNI_ABORT: 不同步数据给jarray,但是会释放inArray
    (*env)->ReleaseIntArrayElements(env, jarray, inArray,0);

}

2. 局部引用和全局引用

  • 1.全局引用
    保存全局变量
jstring globalStr;
JNIEXPORT void  JNICALL Java_com_Amn_ndk_03_Simple_saveGlobalRef
(JNIEnv *env, jclass jclz,jstring str){

    //保存全局变量,方便以后其他方法用到
    globalStr = (*env)->NewGlobalRef(env,str);  
}

不用的时候删除全局变量
(*env)->DeleteGlobalRef(env, globalStr);

JNIEXPORT void  JNICALL Java_com_Amn_ndk_03_Simple_deleteGlobalRef
(JNIEnv *env, jclass jclz){

    (*env)->DeleteGlobalRef(env, globalStr);
}
  • 2.局部引用
    构建的jobject对象若不再使用了 自己要回收,回收之后不能在使用(静态不需要回收)
    (*env)->DeleteLocalRef(env,point);

JNIEXPORT jobject JNICALL java_com_amn_JniSimple_createPoint
(JNIEnv *env, jclass jclz){


    //获取Point 的class  FindClass(env,"全类名") 注: 全类名要反斜杠
    jclass point_clazz = (*env)->FindClass(env, "com/amn/Point");

    //构建函数id    jmethodID j_md = (*env)->GetMethodID(env,point_clazz,"方法名","签名");
    jmethodID j_md = (*env)->GetMethodID(env, point_clazz, "<init>", "(II)V");

    //构建java层的Point 对象 (*env)->NewObject(env, "构建对象的class","构造函数的id",函数参数)
    jobject point = (*env)->NewObject(env, point_clazz, j_md, 10, 20);

    //构建的jobject对象若不再使用了 自己要回收,回收之后不能在使用(静态不需要回收)
    (*env)->DeleteLocalRef(env,point);

}

3.缓存策略

*局部静态缓存 :缓存策略 static
如:native层有一大堆方法要去获取 name 属性

JNIEXPORT void  JNICALL Java_com_Amn_ndk_03_Simple_staticLocalCache
(JNIEnv *env, jclass jclz,jstring name){

    static jfieldID f_id = NULL;//局部缓存,如果这个方法多次被调用,不许要反复去获取
    //获取f_id
    if (f_id ==NULL){
        f_id = (*env)->GetStaticFieldID(env, jclz, "name", " Ljava / lang / String;");
    }
    //name赋值操作
    (*env)->SetStaticObjectField(env,jclz,f_id,name);
}
  • 全局静态缓存 ,在java构造函数中初始化的时候去调用缓存方法
//全局静态缓存 ,在构造函数中初始化的时候去缓存

static jfieldID f_name_id1= NULL;
static jfieldID f_name_id2 = NULL;
static jfieldID f_name_id3 = NULL;


JNIEXPORT void  JNICALL Java_com_Amn_ndk_03_Simple_staticLocalCache
(JNIEnv *env, jclass jclz,jstring name){

    //如果这个方法会反复的调用,那么不会反复的获取jfieldID
    (*env)->SetStaticObjectField(env, jclz, f_name_id1, name);
    
}



JNIEXPORT void  JNICALL Java_com_Amn_ndk_03_Simple_initLocalCache
(JNIEnv *env, jclass jclz){
    //初始化全局静态缓存
    f_name_id1 = (*env)->GetStaticFieldID(env, jclz, "name", " Ljava / lang / String;");
    f_name_id2 = (*env)->GetStaticFieldID(env, jclz, "name", " Ljava / lang / String;");
    f_name_id3 = (*env)->GetStaticFieldID(env, jclz, "name", " Ljava / lang / String;");

}

相关文章

网友评论

      本文标题:JNI基础-缓存策略

      本文链接:https://www.haomeiwen.com/subject/hajbhctx.html