有關(guān)Android JNI開(kāi)發(fā)中比較強大和有用的功能就是從JNI層創(chuàng )建、構造Java的類(lèi)或執行Java層的方法獲取屬性等操作。
一、類(lèi)的相關(guān)操作
1. jclass FindClass(JNIEnv *env, const char *name); 查找類(lèi)
該函數可能做過(guò)Java開(kāi)發(fā)的不會(huì )陌生,這個(gè)是JNI層的實(shí)現,需要注意的是第二個(gè)參數為constchar*類(lèi)型的,我們如果從Java從層傳入unicode編碼的jstring類(lèi)型需要使用GetStringUTFChars函數轉換成utf8的const char*,如果成功返回這個(gè)Java類(lèi)的對象jclass,相關(guān)的異??赡苡?/p>
(1. ClassFormatError 類(lèi)的數據格式無(wú)效
(2. ClassCircularityError 該類(lèi)或接口是自身的超類(lèi)或超接口
(3. NoClassDefFoundError 沒(méi)有找到指定名稱(chēng)的類(lèi)或接口
(4. OOM內存不足錯誤,即OutOfMemoryError
2. jclass GetSuperclass(JNIEnv *env, jclass clazz); 獲取父類(lèi)或者說(shuō)超類(lèi)
該函數的第二個(gè)參數為jclass類(lèi),我們調用時(shí)傳入的是子類(lèi),否則返回將是NULL
3. jboolean IsAssignableFrom(JNIEnv *env, jclass clazz1,jclassclazz2); 判斷class1對象能否安全的強制轉換為class2對象
如果可以將返回 JNI_TRUE,JNI_TRUE的定義值為1,否則返回JNI_FALSE即0 ,這里我們詳細說(shuō)明下哪些情況可能返回真:
(1 這兩個(gè)類(lèi)參數引用同一個(gè) Java 類(lèi)
(2 第一個(gè)類(lèi)是第二個(gè)類(lèi)的子類(lèi)
(3 第二個(gè)類(lèi)是第一個(gè)類(lèi)的某個(gè)接口
4. jclass GetObjectClass(JNIEnv *env, jobject obj); 通過(guò)對象獲取這個(gè)類(lèi)
該函數比較簡(jiǎn)單,唯一注意的是對象不能為NULL,否則獲取的class肯定返回也為NULL。
5. jboolean IsInstanceOf(JNIEnv *env, jobject obj,jclass clazz);判斷對象是否為某個(gè)類(lèi)的實(shí)例
這個(gè)函數是JNI層的實(shí)現,相信大家都不陌生,提醒大家需要注意的是返回值可能產(chǎn)生異議,就是如果傳入的第二個(gè)參數為NULL對象,NULL對象可以強制轉換為各種類(lèi),所以這種情況也將會(huì )返回JNI_TRUE,所以一定判斷傳入的對象是否為空。
6. jboolean IsSameObject(JNIEnv *env, jobject ref1,jobject ref2);判斷兩個(gè)對象是否引用同一個(gè)類(lèi)
需要注意的是如果兩個(gè)對象均為空,返回的值也會(huì )是JNI_TRUE所以使用時(shí)判斷對象為空。
二、調用Java方法
首先說(shuō)下有關(guān)簽名sig相關(guān)的比如 "Ljava/lang/String;"
1. jmethodID GetMethodID(JNIEnv *env, jclass clazz,const char *name,const char *sig); 獲取一個(gè)Java方法的ID
這個(gè)函數將返回非靜態(tài)類(lèi)或接口實(shí)例方法的方法 ID。這個(gè)方法可以是某個(gè)clazz 的超類(lèi)中定義,也可從clazz繼承,最后一個(gè)參數為簽名,最后兩個(gè)參數是constchar*類(lèi)型,是utf8類(lèi)型。需要注意的是,大家執行GetMethodID()函數將導致未初始化的類(lèi)初始化,如果要獲得構造函數的方法ID,使用
(1 NoSuchMethodError 找不到指定的Java方法。
(2 ExceptionInInitializerError 如果由于異常而導致類(lèi)初始化程序失敗
(3 OutOfMemoryError 內存不足
2 . NativeType CallXXXMethod (JNIEnv *env, jobject obj,jmethodIDmethodID, va_list args); 調用XXX類(lèi)型的Java方法
執行Java類(lèi)中某個(gè)方法,需要注意的是這個(gè)里的java類(lèi)是非靜態(tài)的,由于Java的方法的類(lèi)型比較多,所以該函數可能有以下幾種形式,如CallObjectMethod,CallBooleanMethod,CallByteMethod,CallCharMethod,CallShortMethod,CallIntMethod,CallLongMethod,CallFloatMethod,CallDoubleMethod和CallVoidMethod,需要注意的是,該函數的第三個(gè)參數為通過(guò)GetMethodID函數獲取的方法ID,最后一個(gè)參數為這個(gè)方法的參數表,最后的va_list宏可以通過(guò)搜索獲取具體的使用方法,這里不再贅述。
3.NativeType CallNonvirtualXXXMethod (JNIEnv *env, jobject obj,jclassclazz, jmethodID methodID, jvalue *args);
CallNonvirtualXXXMethod函數和上面的CallXXXMethod不同之處是多了一個(gè)jclass參數,CallXXXMethod是根據對象來(lái)調用方法,而CallNonvirtualXXXMethod是根據類(lèi)的實(shí)例調用,區別在這點(diǎn)。
上面的三個(gè)均為非靜態(tài)類(lèi)的獲取,執行調用,需要實(shí)例化這個(gè)類(lèi)才可以執行,下面的為靜態(tài)調用。
4. jmethodID GetStaticMethodID(JNIEnv *env, jclass clazz,const char*name, const char *sig);
5. NativeType CallStaticXXXMethod(JNIEnv *env, jclass clazz,jmethodIDmethodID, ...);
三、訪(fǎng)問(wèn)Java對象的域
Java對象的域或者說(shuō)字段、屬性(Field) 類(lèi)似方法的執行
1. jfieldID GetFieldID(JNIEnv *env, jclass clazz,const char *name,const char *sig); 獲取實(shí)例對象的域ID
需要注意的是,非靜態(tài)的實(shí)例化后的對象,可能產(chǎn)生的異常有
(1 NoSuchFieldError 找不到指定的域
(2 ExceptionInInitializerError 因為異常而導致類(lèi)初始化失敗
(3 OutOfMemoryError內存不足。
2. NativeType GetXXXField(JNIEnv *env, jobject obj,jfieldID fieldID);
類(lèi)似GetXXXMethod函數,可能有的類(lèi)型有GetObjectField,GetBooleanField,GetByteField,GetCharField,GetShortField,GetIntField,GetLongField,GetFloatField,GetDoubleField。
3. void SetXXXField(JNIEnv *env, jobject obj, jfieldIDfieldID,NativeType value);
Java的域可以賦值的,可能有的類(lèi)型有SetObjectField,SetBooleanField,SetByteField,SetCharField,SetShortField,SetIntField,SetLongField,SetFloatField,SetDoubleField。
上面3種情況均為非靜態(tài)對象的域,對于不需要實(shí)例化對象的域,可以直接使用下面的。
4. jfieldID GetStaticFieldID(JNIEnv *env, jclass clazz,const char*name, const char *sig);
5. NativeType GetStaticXXXField(JNIEnv *env, jclass clazz,jfieldIDfieldID);
6. void SetStaticXXXField(JNIEnv *env, jclass clazz,jfieldID fieldID,NativeType value).
聯(lián)系客服