JNI 操作 java 類與對象

jni 操作 java 類

  1. DefineClass

    jclass DefineClass(const char *name, jobject loader, const jbyte* buf,jsize bufLen)
    { return functions->DefineClass(this, name, loader, buf, bufLen); }
    

    從原始類數據的緩沖區加載類;其中 loader 表示類加載器,buf 表示包含 .class 文件數據的緩沖區,bufLen 表示緩沖區長度;

  2. FindClass

    jclass FindClass(const char* name)
    { return functions->FindClass(this, name); }
    

    該函數用于加載本地定義的類;它將搜索由CLASSPATH 環境變量為具有指定名稱的類所指定的目錄和 zip文件; 其中 name 表示類全名;如果 name 以“[“打頭,則返回一個數組類;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        const char* classname = "java/lang/String";
        jclass class_string = env->FindClass(classname);
        if(class_string == NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "FindClass error");
        }else {
            __android_log_print(ANDROID_LOG_INFO, "native", "FindClass success");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  3. GetSuperclass

    jclass GetSuperclass(jclass clazz)
    { return functions->GetSuperclass(this, clazz); }
    

    如果參數 clazz 是任何非 Object 類,返回父類;如果 clazz 是 Object 或接口則返回 NULL;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        const char* classname = "java/lang/String";
        jclass class_string = env->FindClass(classname);
        if(class_string == NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "FindClass error");
        }else {
            __android_log_print(ANDROID_LOG_INFO, "native", "FindClass success");
            jclass class_string_super = env->GetSuperclass(class_string);
            if(class_string_super == NULL){
                __android_log_print(ANDROID_LOG_INFO, "native", "GetSuperclass error");
            }else {
                __android_log_print(ANDROID_LOG_INFO, "native", "GetSuperclass success");
            }
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  4. IsAssignableFrom

    jboolean IsAssignableFrom(jclass clazz1, jclass clazz2)
    { return functions->IsAssignableFrom(this, clazz1, clazz2); }
    

    返回參數類 clazz1 是否可以安全轉換至參數類 clazz2;

    返回為 true 的三種情況:

    1. 類1 與類 2 為同一類;
    2. 類1 是 類2 的子類;
    3. 類1 實現了 類2 實現的某個接口;
    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        const char* string_class_name = "java/lang/String";
        jclass class_string = env->FindClass(string_class_name);
        const char* object_class_name = "java/lang/Object";
        jclass class_object = env->FindClass(object_class_name);
        jboolean isAssignable = env->IsAssignableFrom(class_string, class_object);
        __android_log_print(ANDROID_LOG_INFO, "native", "isAssignable: %d", isAssignable);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    

jni 操作 java 對象

  1. AllocObject

    jobject AllocObject(jclass clazz)
    { return functions->AllocObject(this, clazz); }
    

    分配并返回一個新的 java 對象,不調用其任何構造函數,注意 clazz 不能為數組類;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        const char* classname = "java/lang/String";
        jclass class_string = env->FindClass(classname);
    
        jobject string_object = env->AllocObject(class_string);
    
        if (string_object != NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "AllocObject success");
        }else {
            __android_log_print(ANDROID_LOG_INFO, "native", "AllocObject error");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  2. NewObject、NewObjectV、NewObjectA

    jobject NewObject(jclass clazz, jmethodID methodID, ...)
    {
     va_list args;
     va_start(args, methodID);
     jobject result = functions->NewObjectV(this, clazz, methodID, args);
     va_end(args);
     return result;
    }
    
    jobject NewObjectV(jclass clazz, jmethodID methodID, va_list args)
    { return functions->NewObjectV(this, clazz, methodID, args); }
    
    jobject NewObjectA(jclass clazz, jmethodID methodID, jvalue* args)
    { return functions->NewObjectA(this, clazz, methodID, args); }
    

    這三個方法都是用于分配并返回一個新的 java 對象,methodID用于表示其構造方法(后續解釋);三個方法的區別在于保存構造方法的參數的方式不一樣;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        const char* classname = "java/lang/String";
        jclass class_string = env->FindClass(classname);
        
        jmethodID string_costruct = env->GetMethodID(class_string , "<init>","()V");
        if (string_costruct != NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID success");
        }else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID error");
        }
        
        jobject string_object = env->NewObject(class_string, string_costruct);
        if (string_object != NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "NewObject success");
        }else {
            __android_log_print(ANDROID_LOG_INFO, "native", "NewObject error");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  3. GetObjectClass

    jclass GetObjectClass(jobject obj)
    { return functions->GetObjectClass(this, obj); }
    

    返回 java 對象對應的 java 類

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass thiz_class = env->GetObjectClass(thiz);
        
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  4. GetObjectRefType

    /* added in JNI 1.6 */
    jobjectRefType GetObjectRefType(jobject obj)
    { return functions->GetObjectRefType(this, obj); }
    

    用于返回 java 對象的引用類型, local、global、weak global;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jobjectRefType ref = env->GetObjectRefType(thiz);
        __android_log_print(ANDROID_LOG_INFO, "native", "object ref: %d", ref);
        
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  5. IsInstanceOf

    jboolean IsInstanceOf(jobject obj, jclass obj)
    { return functions->IsInstanceOf(this, obj, clazz); }
    

    返回參數對象 obj 是否參數類 obj 的實例;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        const char* classname = "java/lang/String";
        jclass class_string = env->FindClass(classname);
    
        jboolean isInstanceOf = env->IsInstanceOf(thiz, class_string);
        __android_log_print(ANDROID_LOG_INFO, "native", "isInstanceOf: %d", isInstanceOf);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  6. IsSameObject

    jboolean IsSameObject(jobject ref1, jobject ref2)
    { return functions->IsSameObject(this, ref1, ref2); }
    

    返回兩個對象是否為同一類的實例

jni 操作成員變量

  1. GetFieldID

    jfieldID GetFieldID(jclass clazz, const char* name, const char* sig)
    { return functions->GetFieldID(this, clazz, name, sig); }
    

    根據參數中提供的(非靜態)成員變量名稱,以及成員變量類型返回 field ID;其中 name 表示成員變量名稱,sig 表示成員變量的域描述符;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
        jfieldID fid= env->GetFieldID(cls, "s", "Ljava/lang/String;");
        if (fid == NULL) {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetFieldID error");
        } else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetFieldID success");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  2. Get<type>Field

    jobject GetObjectField(jobject obj, jfieldID fieldID)
    { return functions->GetObjectField(this, obj, fieldID); }
    jboolean GetBooleanField(jobject obj, jfieldID fieldID)
    { return functions->GetBooleanField(this, obj, fieldID); }
    jbyte GetByteField(jobject obj, jfieldID fieldID)
    { return functions->GetByteField(this, obj, fieldID); }
    jchar GetCharField(jobject obj, jfieldID fieldID)
    { return functions->GetCharField(this, obj, fieldID); }
    jshort GetShortField(jobject obj, jfieldID fieldID)
    { return functions->GetShortField(this, obj, fieldID); }
    jint GetIntField(jobject obj, jfieldID fieldID)
    { return functions->GetIntField(this, obj, fieldID); }
    jlong GetLongField(jobject obj, jfieldID fieldID)
    { return functions->GetLongField(this, obj, fieldID); }
    __NDK_FPABI__
    jfloat GetFloatField(jobject obj, jfieldID fieldID)
    { return functions->GetFloatField(this, obj, fieldID); }
    __NDK_FPABI__
    jdouble GetDoubleField(jobject obj, jfieldID fieldID)
    { return functions->GetDoubleField(this, obj, fieldID); }
    

    根據參數中的 fieldID 獲取參數對象 obj 中的成員變量;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
        jfieldID fid= env->GetFieldID(cls, "s", "Ljava/lang/String;");
        if (fid == NULL) {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetFieldID error");
        } else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetFieldID success");
        }
        jstring jstr = (jstring) env->GetObjectField(thiz, fid);
        const char* str = env->GetStringUTFChars(jstr, NULL);
        __android_log_print(ANDROID_LOG_INFO, "native", "GetObjectField string: %s", str);
        
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  3. Set<type>Filed

    void SetObjectField(jobject obj, jfieldID fieldID, jobject value)
    { functions->SetObjectField(this, obj, fieldID, value); }
    void SetBooleanField(jobject obj, jfieldID fieldID, jboolean value)
    { functions->SetBooleanField(this, obj, fieldID, value); }
    void SetByteField(jobject obj, jfieldID fieldID, jbyte value)
    { functions->SetByteField(this, obj, fieldID, value); }
    void SetCharField(jobject obj, jfieldID fieldID, jchar value)
    { functions->SetCharField(this, obj, fieldID, value); }
    void SetShortField(jobject obj, jfieldID fieldID, jshort value)
    { functions->SetShortField(this, obj, fieldID, value); }
    void SetIntField(jobject obj, jfieldID fieldID, jint value)
    { functions->SetIntField(this, obj, fieldID, value); }
    void SetLongField(jobject obj, jfieldID fieldID, jlong value)
    { functions->SetLongField(this, obj, fieldID, value); }
    __NDK_FPABI__
    void SetFloatField(jobject obj, jfieldID fieldID, jfloat value)
    { functions->SetFloatField(this, obj, fieldID, value); }
    __NDK_FPABI__
    void SetDoubleField(jobject obj, jfieldID fieldID, jdouble value)
    { functions->SetDoubleField(this, obj, fieldID, value); }
    

    根據參數中的 fieldID 設置參數對象 obj 中的成員變量;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
        jfieldID fid= env->GetFieldID(cls, "s", "Ljava/lang/String;");
        if (fid == NULL) {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetFieldID error");
        } else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetFieldID success");
        }
        jstring jstr = env->NewStringUTF("world");
        env->SetObjectField(thiz, fid, jstr);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    

jni 操作成員方法

  1. GetMethodID

    jmethodID GetMethodID(jclass clazz, const char* name, const char* sig)
    { return functions->GetMethodID(this, clazz, name, sig); }
    

    根據參數中提供的(非靜態)成員方法名稱,以及方法參數與返回值獲取 method ID;其中 name 表示成員方法名稱,sig 表示成員方法參數與返回值;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
    
        jmethodID mid = env->GetMethodID(cls, "printString", "()V");
        if (mid == NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID error ");
        }else{
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID success");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  2. Call<type>Method、Call<type>MethodA、Call<type>MethodV

    三個方法都是用于調用對象的成員方法,區別在于方法參數的傳入方式不一樣,其中<type>用于表示返回值類型;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
    
        jmethodID mid = env->GetMethodID(cls, "printString", "()V");
        if (mid == NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID error ");
        }else{
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID success");
        }
        env->CallVoidMethod(thiz, mid);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  3. CallNonvirtual<type>Method、CallNonvirtual<type>MethodA、CallNonvirtual<type>MethodV

    這三個方法也是用于調用對象的成員方法,與上述三個方法不同的是:Call<type>Method 是基于對象調用方法,而 CallNonvirtual<type>Method 是基于類調用方法,也就是說 CallNonvirtual<type>Method 可以調用參數對象的父類并非必須當前類;

jni 調用靜態成員變量

  1. GetStaticFieldID

    jfieldID GetStaticFieldID(jclass clazz, const char* name, const char* sig)
    { return functions->GetStaticFieldID(this, clazz, name, sig); }
    

    根據參數中提供的(靜態)成員變量名稱,以及成員變量類型返回 field ID;其中 name 表示成員變量名稱,sig 表示成員變量的域描述符;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
        jfieldID fid= env->GetStaticFieldID(cls, "s", "Ljava/lang/String;");
        if (fid == NULL) {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetStaticFieldID error");
        } else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetStaticFieldID success");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  2. GetStatic<type>Field

    jobject GetStaticObjectField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticObjectField(this, clazz, fieldID); }
    jboolean GetStaticBooleanField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticBooleanField(this, clazz, fieldID); }
    jbyte GetStaticByteField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticByteField(this, clazz, fieldID); }
    jchar GetStaticCharField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticCharField(this, clazz, fieldID); }
    jshort GetStaticShortField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticShortField(this, clazz, fieldID); }
    jint GetStaticIntField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticIntField(this, clazz, fieldID); }
    jlong GetStaticLongField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticLongField(this, clazz, fieldID); }
    __NDK_FPABI__
    jfloat GetStaticFloatField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticFloatField(this, clazz, fieldID); }
    __NDK_FPABI__
    jdouble GetStaticDoubleField(jclass clazz, jfieldID fieldID)
    { return functions->GetStaticDoubleField(this, clazz, fieldID); }
    

    根據參數中的 fieldID 獲取參數類 clazz 中的成員變量;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
        jfieldID fid= env->GetStaticFieldID(cls, "s", "Ljava/lang/String;");
        if (fid == NULL) {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetStaticFieldID error");
        } else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetStaticFieldID success");
        }
        jstring jstr = (jstring) env->GetStaticObjectField(cls, fid);
        
        const char* str = env->GetStringUTFChars(jstr, NULL);
        __android_log_print(ANDROID_LOG_INFO, "native", "GetStringUTFChars %s", str);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  3. SetStatic<type>Field

    void SetStaticObjectField(jclass clazz, jfieldID fieldID, jobject value)
    { functions->SetStaticObjectField(this, clazz, fieldID, value); }
    void SetStaticBooleanField(jclass clazz, jfieldID fieldID, jboolean value)
    { functions->SetStaticBooleanField(this, clazz, fieldID, value); }
    void SetStaticByteField(jclass clazz, jfieldID fieldID, jbyte value)
    { functions->SetStaticByteField(this, clazz, fieldID, value); }
    void SetStaticCharField(jclass clazz, jfieldID fieldID, jchar value)
    { functions->SetStaticCharField(this, clazz, fieldID, value); }
    void SetStaticShortField(jclass clazz, jfieldID fieldID, jshort value)
    { functions->SetStaticShortField(this, clazz, fieldID, value); }
    void SetStaticIntField(jclass clazz, jfieldID fieldID, jint value)
    { functions->SetStaticIntField(this, clazz, fieldID, value); }
    void SetStaticLongField(jclass clazz, jfieldID fieldID, jlong value)
    { functions->SetStaticLongField(this, clazz, fieldID, value); }
    __NDK_FPABI__
    void SetStaticFloatField(jclass clazz, jfieldID fieldID, jfloat value)
    { functions->SetStaticFloatField(this, clazz, fieldID, value); }
    __NDK_FPABI__
    void SetStaticDoubleField(jclass clazz, jfieldID fieldID, jdouble value)
    { functions->SetStaticDoubleField(this, clazz, fieldID, value); }
    

    根據參數中的 fieldID 設置參數類 clazz 中的成員變量;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
        jfieldID fid= env->GetStaticFieldID(cls, "s", "Ljava/lang/String;");
        if (fid == NULL) {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetStaticFieldID error");
        } else {
            __android_log_print(ANDROID_LOG_INFO, "native", "GetStaticFieldID success");
        }
        jstring jstr = env->NewStringUTF("world");
        env->SetStaticObjectField(cls, fid, jstr);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    

jni 操作靜態方法

  1. GetStaticMethodID

    jmethodID GetStaticMethodID(jclass clazz, const char* name, const char* sig)
    { return functions->GetStaticMethodID(this, clazz, name, sig); }
    

    根據參數中提供的(靜態)成員方法名稱,以及方法參數與返回值獲取 method ID;其中 name 表示成員方法名稱,sig 表示成員方法參數與返回值;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
    
        jmethodID mid = env->GetStaticMethodID(cls, "printString", "()V");
        if (mid == NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID error ");
        }else{
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID success");
        }
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
  2. CallStatic<type>Method、CallStatic<type>MethodV、CallStatic<type>MethodA

    三個方法都是用于調用類的靜態方法,區別在于方法參數的傳入方式不一樣,其中<type>用于表示返回值類型;

    jstring stringFromJNI(JNIEnv *env, jobject thiz){
        __android_log_print(ANDROID_LOG_INFO, "native", "stringFromJNI");
    
        jclass cls = env->GetObjectClass(thiz);
    
        jmethodID mid = env->GetStaticMethodID(cls, "printString", "()V");
        if (mid == NULL){
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID error ");
        }else{
            __android_log_print(ANDROID_LOG_INFO, "native", "GetMethodID success");
        }
    
        env->CallStaticVoidMethod(cls, mid);
    
        std::string hello = "Hello from C++";
        return env->NewStringUTF(hello.c_str());
    }
    
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。