首先说说在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);
}