JNI中的基本内容(生成头文件&方法,属性签名方法)

首先说说在JAVA中写了一个NATIVE方法,然后生成头文件的方法

javah -classpath java生在目录 -d 同左 -jni 包名+类名

C:\Users\baketsCat>javah -classpath D:\project\Andfix\app\src\main\java -d D:\

oject\Andfix\app\src\main\java -jni com.example.baketscat.andfix.PosixThread

如上,就可以在JAVA目录生成一个头文件了!

由于在JNI中,经常需要方法的签名,或者属性的签名,当然,可以根据签名规则来写签名,但太容易错了。这里提供一种通用方法来查看方法&属性的签名:

找签名

首先在CMD中

cd /D D:

cd D:\project\Ffmpeg\app\build\intermediates\classes\debug\com\example\baketscat\ffmpeg

对,这个路径!是在build---debug中 找到对应的class上一层文件,比如说要查看VideoUtils这个native类,那么就要在找到VideoUtils的包文件的路径

然后!

javap -s VideoUtils.class

VideoUtils.class就是native类,然后就会看到这个类里面,所有方法,所有变量的签名了!!



下面讲一些非常基本的JNI方法。

JNI方法中,至少会有两个参数 JNIEnv * env, jobject obj,注意,jobject这里是类为非静态类,若为静态类,则此处为jclass

//得到class

jclass cls = (*env)->GetObjectClass(env, obj);

//jfieldID   获得一个JAVA变量

//签名:类型的简称

//属性,方法

jfieldID fid = (*env)->GetFieldID(env, cls, "key", "Ljava/lang/String;");

//获取key属性的值//注意:key为基本数据类型,规则如下//(*env)->GetIntField(); (*env)->GetField();

jstring jstr = (*env)->GetObjectField(env, obj, fid);

//jstring转为 C/C++字符串

char *str = (*env)->GetStringUTFChars(env, jstr, NULL);

//C字符呺 转为j

//拼接字符串

char text[50] = "super ";

strcat(text,str);

//拼接完成之后,从C字符串转为

jstringjstr = (*env)->NewStringUTF(env, text);

//修改key的属性//注意规则:SetField

(*env)->SetObjectField(env, obj, fid, jstr);

//访问静态变量

jclass cls = (*env)->GetObjectClass(env, obj);

jfieldID fid = (*env)->GetStaticFieldID(env, cls, "count", "I");

//获取静态属性的值//规则:

GetStaticFieldjint count = (*env)->GetStaticIntField(env, cls, fid);count += 10;

//修改属性的值//规则:SetStaticField   调用后,JAVA的静变也会改变

(*env)->SetStaticIntField(env, cls, fid, count);

//C调用JAVA非静态方法,并获得JAVA方法的返回值

jclass cls = (*env)->GetObjectClass(env, obj);//jmethodID

jmethodID mid = (*env)->GetMethodID(env, cls, "genRandomInt", "(I)I");

//调用方法,产生了一个随机数//规则:CallMethod 返回值类型

//这里就是调用一个返回值为INT的方法!

jint random = (*env)->CallIntMethod(env, obj, mid, 200);

//C调JAVA静态方法

//如果native方法为static,jobject为子类jclass的实例,也就是native方法所属的类的Class实例

//所以,这里的cls,是jobject cls 而不是 jclass cls

//jclass  

//jclass cls = (*env)->GetObjectClass(env, obj);

//jmethodID

jmethodID mid = (*env)->GetStaticMethodID(env, cls, "getUUID", "()Ljava/lang/String;");

//调用//规则:CallStaticMethod

jstring uuid = (*env)->CallStaticObjectMethod(env, cls, mid);

//jstring转为C字符串

char *uuid_str = (*env)->GetStringUTFChars(env, uuid, NULL);

//访问构造方法

//目的是能够访问JAVA类!!

//Date jclassjclass cls = (*env)->FindClass(env, "java/util/Date");

//构造方法jmethodID

jmethodID contructor_mid = (*env)->GetMethodID(env, cls, "","()V");

//实例化一个Date对象

jobject date_obj = (*env)->NewObject(env, cls, contructor_mid);

//调用getTime方法

jmethodID mid = (*env)->GetMethodID(env, cls, "getTime", "()J");

jlong time = (*env)->CallLongMethod(env, date_obj, mid);


JAVA中的NATIVE方法,传入一个INT数据到C

参数如下:JNIEnv * env, jobject obj, jintArray arr

//Java的int数组(jintArray)->C int数组

jint *elems = (*env)->GetIntArrayElements(env, arr, NULL);

//数组的长度

int len = (*env)->GetArrayLength(env, arr);

//对(jint)long数组进行

qsort(elems, len, sizeof(jint), compare);

//同步

//释放数组的元素

//mode参数

//0,Java数组进行更新,并且释放C/C++数组

//JNI_ABORT,Java数组不进行更新,但是释放C/C++数组

//JNI_COMMIT,Java数组进行更新,不释放C/C++数组(函数执行完,数组还是会释放)

(*env)->ReleaseIntArrayElements(env, arr, elems, JNI_COMMIT);

关键是同步!就是说,让JAVA的数组也相应变化!

//局部引用

//局部引用会在C/C++代码执行完成之后自动释放(可以回收)

//但是,有时候我们需要手动去释放

//1.访问一个很大的Java对象,使用完之后,还要进行复杂的耗时操作

//2.创建了大量的局部引用,占用了太多的内存,而且这些布局引用跟后面的操作没有关联性

int i = 0;

for (; i < 5; i++){

   jclass cls = (*env)->FindClass(env, "java/util/Date");

   jmethodID constructor_mid = (*env)->GetMethodID(env, cls, "", "()V");

   //实例化Date对象

   jobject obj = (*env)->NewObject(env, cls, constructor_mid);

   //Date对象数组

   jobjectArray jobj_arr = (*env)->NewObjectArray(env, 5, cls, obj);

   //提前释放,不要占用内存太久

   //告诉虚拟机垃圾回收器,可以回收这些对象

   (*env)->DeleteLocalRef(env, obj);

   (*env)->DeleteLocalRef(env, jobj_arr);

}

//注意:局部引用不能在多个线程间传递

//全局引用

//可以跨越多个线程,在程序员手动释放之前,一直有效


jstring global_str;

//设置global_str

JNIEXPORT void JNICALL Java_com_tz_jni_TestNative_createGlobalRef(JNIEnv * env, jobject j_obj){

   jstring obj = (*env)->NewStringUTF(env, "jni development is powerful!");

   global_str = (*env)->NewGlobalRef(env, obj);

}

//访问global_str

JNIEXPORT jstring JNICALL Java_com_tz_jni_TestNative_getGlobalRef(JNIEnv * env, jobject obj){

  return global_str;

}

//释放global_str

JNIEXPORT void JNICALL Java_com_tz_jni_TestNative_deleteGlobalRef(JNIEnv * env, jobject obj){

  (*env)->DeleteGlobalRef(env, global_str);

}

你可能感兴趣的:(JNI中的基本内容(生成头文件&方法,属性签名方法))