首先說說在JAVA中寫了一個(gè)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目錄生成一個(gè)頭文件了!
由于在JNI中,經(jīng)常需要方法的簽名,或者屬性的簽名,當(dāng)然,可以根據(jù)簽名規(guī)則來寫簽名,但太容易錯(cuò)了。這里提供一種通用方法來查看方法&屬性的簽名:
找簽名
首先在CMD中
cd /D D:
cd D:\project\Ffmpeg\app\build\intermediates\classes\debug\com\example\baketscat\ffmpeg
對(duì),這個(gè)路徑!是在build---debug中 找到對(duì)應(yīng)的class上一層文件,比如說要查看VideoUtils這個(gè)native類,那么就要在找到VideoUtils的包文件的路徑
然后!
javap -s VideoUtils.class
VideoUtils.class就是native類,然后就會(huì)看到這個(gè)類里面,所有方法,所有變量的簽名了!!
下面講一些非常基本的JNI方法。
JNI方法中,至少會(huì)有兩個(gè)參數(shù) JNIEnv * env, jobject obj,注意,jobject這里是類為非靜態(tài)類,若為靜態(tài)類,則此處為jclass
//得到class
jclass cls = (*env)->GetObjectClass(env, obj);
//jfieldID ? 獲得一個(gè)JAVA變量
//簽名:類型的簡(jiǎn)稱
//屬性,方法
jfieldID fid = (*env)->GetFieldID(env, cls, "key", "Ljava/lang/String;");
//獲取key屬性的值//注意:key為基本數(shù)據(jù)類型,規(guī)則如下//(*env)->GetIntField(); (*env)->GetField();
jstring jstr = (*env)->GetObjectField(env, obj, fid);
//jstring轉(zhuǎn)為 C/C++字符串
char *str = (*env)->GetStringUTFChars(env, jstr, NULL);
//C字符呺 轉(zhuǎn)為j
//拼接字符串
char text[50] = "super ";
strcat(text,str);
//拼接完成之后,從C字符串轉(zhuǎn)為
jstringjstr = (*env)->NewStringUTF(env, text);
//修改key的屬性//注意規(guī)則:SetField
(*env)->SetObjectField(env, obj, fid, jstr);
//訪問靜態(tài)變量
jclass cls = (*env)->GetObjectClass(env, obj);
jfieldID fid = (*env)->GetStaticFieldID(env, cls, "count", "I");
//獲取靜態(tài)屬性的值//規(guī)則:
GetStaticFieldjint count = (*env)->GetStaticIntField(env, cls, fid);count += 10;
//修改屬性的值//規(guī)則:SetStaticField ? 調(diào)用后,JAVA的靜變也會(huì)改變
(*env)->SetStaticIntField(env, cls, fid, count);
//C調(diào)用JAVA非靜態(tài)方法,并獲得JAVA方法的返回值
jclass cls = (*env)->GetObjectClass(env, obj);//jmethodID
jmethodID mid = (*env)->GetMethodID(env, cls, "genRandomInt", "(I)I");
//調(diào)用方法,產(chǎn)生了一個(gè)隨機(jī)數(shù)//規(guī)則:CallMethod 返回值類型
//這里就是調(diào)用一個(gè)返回值為INT的方法!
jint random = (*env)->CallIntMethod(env, obj, mid, 200);
//C調(diào)JAVA靜態(tài)方法
//如果native方法為static,jobject為子類jclass的實(shí)例,也就是native方法所屬的類的Class實(shí)例
//所以,這里的cls,是jobject cls 而不是 jclass cls
//jclass ?
//jclass cls = (*env)->GetObjectClass(env, obj);
//jmethodID
jmethodID mid = (*env)->GetStaticMethodID(env, cls, "getUUID", "()Ljava/lang/String;");
//調(diào)用//規(guī)則:CallStaticMethod
jstring uuid = (*env)->CallStaticObjectMethod(env, cls, mid);
//jstring轉(zhuǎn)為C字符串
char *uuid_str = (*env)->GetStringUTFChars(env, uuid, NULL);
//訪問構(gòu)造方法
//目的是能夠訪問JAVA類!!
//Date jclassjclass cls = (*env)->FindClass(env, "java/util/Date");
//構(gòu)造方法jmethodID
jmethodID contructor_mid = (*env)->GetMethodID(env, cls, "","()V");
//實(shí)例化一個(gè)Date對(duì)象
jobject date_obj = (*env)->NewObject(env, cls, contructor_mid);
//調(diào)用getTime方法
jmethodID mid = (*env)->GetMethodID(env, cls, "getTime", "()J");
jlong time = (*env)->CallLongMethod(env, date_obj, mid);
JAVA中的NATIVE方法,傳入一個(gè)INT數(shù)據(jù)到C
參數(shù)如下:JNIEnv * env, jobject obj, jintArray arr
//Java的int數(shù)組(jintArray)->C int數(shù)組
jint *elems = (*env)->GetIntArrayElements(env, arr, NULL);
//數(shù)組的長度
int len = (*env)->GetArrayLength(env, arr);
//對(duì)(jint)long數(shù)組進(jìn)行
qsort(elems, len, sizeof(jint), compare);
//同步
//釋放數(shù)組的元素
//mode參數(shù)
//0,Java數(shù)組進(jìn)行更新,并且釋放C/C++數(shù)組
//JNI_ABORT,Java數(shù)組不進(jìn)行更新,但是釋放C/C++數(shù)組
//JNI_COMMIT,Java數(shù)組進(jìn)行更新,不釋放C/C++數(shù)組(函數(shù)執(zhí)行完,數(shù)組還是會(huì)釋放)
(*env)->ReleaseIntArrayElements(env, arr, elems, JNI_COMMIT);
關(guān)鍵是同步!就是說,讓JAVA的數(shù)組也相應(yīng)變化!
//局部引用
//局部引用會(huì)在C/C++代碼執(zhí)行完成之后自動(dòng)釋放(可以回收)
//但是,有時(shí)候我們需要手動(dòng)去釋放
//1.訪問一個(gè)很大的Java對(duì)象,使用完之后,還要進(jìn)行復(fù)雜的耗時(shí)操作
//2.創(chuàng)建了大量的局部引用,占用了太多的內(nèi)存,而且這些布局引用跟后面的操作沒有關(guān)聯(lián)性
int i = 0;
for (; i < 5; i++){
? ?jclass cls = (*env)->FindClass(env, "java/util/Date");
? ?jmethodID constructor_mid = (*env)->GetMethodID(env, cls, "", "()V");
? ?//實(shí)例化Date對(duì)象
? ?jobject obj = (*env)->NewObject(env, cls, constructor_mid);
? ?//Date對(duì)象數(shù)組
? ?jobjectArray jobj_arr = (*env)->NewObjectArray(env, 5, cls, obj);
? ?//提前釋放,不要占用內(nèi)存太久
? ?//告訴虛擬機(jī)垃圾回收器,可以回收這些對(duì)象
? ?(*env)->DeleteLocalRef(env, obj);
? ?(*env)->DeleteLocalRef(env, jobj_arr);
}
//注意:局部引用不能在多個(gè)線程間傳遞
//全局引用
//可以跨越多個(gè)線程,在程序員手動(dòng)釋放之前,一直有效
jstring global_str;
//設(shè)置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);
}