NDK开发 - C/C++ 访问 Java 变量和方法

字节行者说
• 阅读 3532

上一篇有提到 JNI 访问引用数组,涉及了 C/C++ 访问 Java 实例的方法和变量。虽然在之前的开发中,并没有用到 C/C++ 范围 Java 层数据,但是这部分内容还是很有用的。

传送门:NDK开发 - C/C++ 访问 Java 变量和方法

  C/C++ 访问 Java 层的方法和变量主要分为实例和静态。调用静态变量和方法,直接通过 类名.方法/类名.变量,而调用实例变量和方法,需要先实例化对象才可以调用。另外对于继承的子类访问父类的方法也是可以通过 JNI 实现,虽然可能用的不多。

调用静态变量和方法

  静态数据的访问其实比较简单。当我们在运行一个 Java 程序时,JVM 会先将程序运行时所要用到所有相关的 class 文件加载到 JVM 中,并采用按需加载的方式加载,也就是说某个类只有在被用到的时候才会被加载,这样设计的目的也是为了提高程序的性能和节约内存。本地代码也是按照上面的流程来访问类的静态方法或实例方法的。下面直接上代码:

Java层代码

 //调用静态方法/变量
 NativeMethod.invokeStaticFieldAndMethod(23, "gnaix");
 Log.d(TAG, "field value:" + Util.STATIC_FIELD);
/**
 * 名称: Util
 * 描述:
 *
 * @author xiangqing.xue
 * @date 16/3/11
 */
public class Util {
    private static String TAG = "Util";

    public static int STATIC_FIELD = 300;

    public static String getStaticMethod(String name, int age){
        String str =  "Hello "+ name + ", age:" + age;
        Log.d(TAG, str);
        return str;
    }
}

Native代码

JNIEXPORT void JNICALL Java_com_example_gnaix_ndk_NativeMethod_invokeStaticFieldAndMethod
        (JNIEnv *env, jclass object, jint age, jstring name){

    jclass clazz = NULL;
    jfieldID fid = NULL;
    jmethodID mid = NULL;
    jstring j_result = NULL;

    //1. 获取Util类的Class引用
    clazz = env->FindClass("com/example/gnaix/ndk/Util");
    if(clazz == NULL){
        LOGD("clazz null");
        return;
    }

    //2. 获取静态变量属性ID和方法ID
    fid = env->GetStaticFieldID(clazz, "STATIC_FIELD", "I");
    if(fid == NULL){
        LOGD("fieldID null");
        return;
    }
    mid = env->GetStaticMethodID(clazz, "getStaticMethod", "(Ljava/lang/String;I)Ljava/lang/String;");
    if(mid == NULL){
        LOGD("method null");
        return;
    }

    //3. 获取静态变量值和调用静态方法
    jint num = env->GetStaticIntField(clazz, fid);
    LOGD("static field value: %d", num);
    j_result = (jstring)env->CallStaticObjectMethod(clazz, mid, name, age);
    const char *buf_result = env->GetStringUTFChars(j_result, 0);
    LOGD("static: %s", buf_result);
    env->ReleaseStringUTFChars(j_result, buf_result);

    //4. 修改静态变量值
    env->SetStaticIntField(clazz, fid, 100);

    //5. 释放局部引用
    env->DeleteLocalRef(clazz);
    env->DeleteLocalRef(j_result);
}

运行结果

NDK开发 - C/C++ 访问 Java 变量和方法

代码解析

第一步:调用 jclass FindClass(const char* name) 参数为 Java 类的 classPath 只是把.换成/,获取jclass。

第二步:获取变量和方法的签名ID
jfieldID GetStaticFieldID(jclass clazz, const char* name, const char* sig)
jmethodID GetStaticMethodID(jclass clazz, const char* name, const char* sig)
参数说明:

  • clazz: Java类的 Class 对象(第一步获取的jclass对象)

  • name: 变量名/方法名

  • sig: 变量名/方法名签名ID

签名获取的方法在上一篇有提到过:NDK开发 - JNI数组数据处理

第三步1:获取静态变量的值
jint GetStaticIntField(jclass clazz, jfieldID fieldID)方法获取int类型静态变量的值,相同的函数还有 GetStaticLongField, GetStaticFloatField, GetStaticDoubleField等,引用类型都是用 GetStaticObjectField

第三步2:调用静态函数,通过 JNI 的 CallStaticObjectMethod(clazz, mid, name, age) 函数调用 Java 的 getStaticMethod 静态函数,第一个参数为 Java 的 Class 对象,mid 为方法的签名, 后面的参数为 getStaticMethod 的参数,返回值为 jstring。同样也有 CallStaticXXMethod 方法调用表示不同的返回值。

第四步:通过 SetStaticIntField改变静态变量的值。

第五步:释放局部变量

调用实例变量和方法

  调用实例变量与方法与范围静态的类型,不过多了一个实例对象的过程。

Java 层代码

 //调用实例方法/变量
 NativeMethod.invokeJobject("gnaix", 23);
/**
 * 名称: Person
 * 描述:
 *
 * @author xiangqing.xue
 * @date 16/3/10
 */
public class Person {
    private String TAG = "Person";

    public String name;
    public int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String toString(){
        String result =  "Name: " + name + ", age: " + age;
        Log.d(TAG, result);
        return result;
    }
}

Native代码

JNIEXPORT void JNICALL Java_com_example_gnaix_ndk_NativeMethod_invokeJobject
        (JNIEnv *env, jclass object, jstring name, jint age){
    jclass clazz = NULL;
    jobject jobj = NULL;
    jfieldID fid_name = NULL;
    jfieldID fid_age = NULL;
    jmethodID mid_construct = NULL;
    jmethodID mid_to_string = NULL;
    jstring j_new_str;
    jstring j_result;

    //1. 获取Person类Class引用
    clazz = env->FindClass("com/example/gnaix/ndk/Person");
    if(clazz == NULL){
        LOGD("clazz null");
        return;
    }

    //2. 获取类的默认构造函数ID
    mid_construct = env->GetMethodID(clazz, "<init>", "()V");
    if(mid_construct == NULL){
        LOGD("construct null");
        return;
    }

    //3. 获取实例方法ID和变量ID
    fid_name = env->GetFieldID(clazz, "name", "Ljava/lang/String;");
    fid_age = env->GetFieldID(clazz, "age", "I");
    mid_to_string = env->GetMethodID(clazz, "toString", "()Ljava/lang/String;");
    if(fid_age==NULL || fid_name==NULL || mid_to_string==NULL){
        LOGD("age|name|toString null");
        return;
    }

    //4. 创建该类的实例
    jobj = env->NewObject(clazz, mid_construct);
    if(jobj == NULL){
        LOGD("jobject null");
        return;
    }

    //5. 修改变量值
    env->SetIntField(jobj, fid_age, 23);
    j_new_str = env->NewStringUTF("gnaix");
    env->SetObjectField(jobj, fid_name, j_new_str);

    //6. 调用实例方法
    j_result = (jstring)env->CallObjectMethod(jobj, mid_to_string);
    const char *buf_result = env->GetStringUTFChars(j_result, 0);
    LOGD("object: %s", buf_result);
    env->ReleaseStringUTFChars(j_result, buf_result);

    //7. 释放局部引用
    env->DeleteLocalRef(clazz);
    env->DeleteLocalRef(jobj);
    env->DeleteLocalRef(j_new_str);
    env->DeleteLocalRef(j_result);
}

运行结果

NDK开发 - C/C++ 访问 Java 变量和方法

代码解析

  访问实例变量和方法没有很大区别多了一个实例化的步骤。每个类会生成一个默认的构造函数,调用的时候用 <init> ,签名ID是 ()V。如果是自定义的构造函数需要用相应的签名ID。

//2. 获取类的默认构造函数ID
mid_construct = env->GetMethodID(clazz, "<init>", "()V");
if(mid_construct == NULL){
    LOGD("construct null");
    return;
}

//4. 创建该类的实例
jobj = env->NewObject(clazz, mid_construct);
if(jobj == NULL){
    LOGD("jobject null");
    return;
}

调用构造方法和父类实例方法

Java 代码

  定义了 Animal 和 Dog 两个类,Animal 定义了一个 String 形参的构造方法,一个成员变量 name、两个成员函数 run 和 getName,Dog 继承自 Animal,并重写了 run 方法。在 JNI 中实现创建 Dog 对象的实例,调用 Animal 类的 run 和 getName 方法。代码如下所示:

/**
 * 名称: Animal
 * 描述:
 *
 * @author xiangqing.xue
 * @date 16/4/1
 */
public class Animal {
    private String TAG = "Animal";

    protected String name;

    public Animal(String name){
        this.name = name;
        Log.d(TAG, "Animal Construct call...");
    }

    public String getName(){
        Log.d(TAG, "Animal.getName Call...");
        return this.name;
    }

    public void run(){
        Log.d(TAG, "Animal.run...");
    }
}
/**
 * 名称: Dog
 * 描述:
 *
 * @author xiangqing.xue
 * @date 16/4/1
 */
public class Dog extends Animal{

    private String TAG = "Dog";

    public Dog(String name) {
        super(name);
        Log.d(TAG, "Dog Construct call...");
    }

    @Override
    public String getName() {
        return "My name is " + this.name;
    }

    @Override
    public void run() {
        Log.d(TAG, name + "Dog.run...");
    }
}

Native 代码

JNIEXPORT void JNICALL Java_com_example_gnaix_ndk_NativeMethod_callSuperInstanceMethod
        (JNIEnv *env, jclass jobj){
    jclass cls_dog;
    jclass cls_animal;
    jmethodID mid_dog_init;
    jmethodID mid_run;
    jmethodID mid_get_name;
    jstring c_str_name;
    jobject obj_dog;
    const char *name = NULL;

    //1. 获取Dog类Class引用
    cls_dog = env->FindClass("com/example/gnaix/ndk/Dog");
    if(cls_dog == NULL){
        LOGD("cls_dog null");
        return;
    }

    //2. 获取Dog类构造方法ID
    mid_dog_init = env->GetMethodID(cls_dog, "<init>", "(Ljava/lang/String;)V");
    if(mid_dog_init == NULL){
        LOGD("cls_dog init null");
        return;
    }

    //3. 创建Dog对象实例
    c_str_name = env->NewStringUTF("Tom");
    obj_dog = env->NewObject(cls_dog, mid_dog_init, c_str_name);
    if(obj_dog == NULL){
        LOGD("obj_dog null");
        return;
    }

    //4. 获取animal类Class引用
    cls_animal = env->FindClass("com/example/gnaix/ndk/Animal");
    if(cls_animal == NULL){
        LOGD("cls_dog null");
        return;
    }

    //5. 获取父类run方法ID并调用
    mid_run = env->GetMethodID(cls_animal, "run", "()V");
    if(mid_run == NULL){
        LOGD("mid_run null");
        return;
    }
    env->CallNonvirtualVoidMethod(obj_dog, cls_animal, mid_run);

    //6. 调用父类getName方法
    mid_get_name = env->GetMethodID(cls_animal, "getName", "()Ljava/lang/String;");
    if(mid_get_name == NULL){
        LOGD("mid_get_name null");
        return;
    }
    c_str_name = (jstring) env->CallNonvirtualObjectMethod(obj_dog, cls_animal, mid_get_name);
    name = env->GetStringUTFChars(c_str_name, 0);
    LOGD("In C: Animal Name is %s", name);

    //7. 释放从java层获取到的字符串所分配的内存
    env->ReleaseStringUTFChars(c_str_name, name);

    //8. 释放局部变量
    env->DeleteLocalRef(cls_animal);
    env->DeleteLocalRef(cls_dog);
    env->DeleteLocalRef(c_str_name);
    env->DeleteLocalRef(obj_dog);
}

运行结果

NDK开发 - C/C++ 访问 Java 变量和方法

代码解析

调用构造方法

调用构造方法和调用对象的实例方法方式是相似的,传入”< init >”作为方法名查找类的构造方法ID,然后调用JNI函数NewObject调用对象的构造函数初始化对象。如下代码所示:

 //2. 获取Dog类构造方法ID
mid_dog_init = env->GetMethodID(cls_dog, "<init>", "(Ljava/lang/String;)V");
if(mid_dog_init == NULL){
    LOGD("cls_dog init null");
    return;
}

//3. 创建Dog对象实例
c_str_name = env->NewStringUTF("Tom");
obj_dog = env->NewObject(cls_dog, mid_dog_init, c_str_name);
if(obj_dog == NULL){
    LOGD("obj_dog null");
    return;
}

调用父类实例方法

  如果一个方法被定义在父类中,在子类中被覆盖,也可以调用父类中的这个实例方法。JNI 提供了一系列函数CallNonvirtualXXXMethod 来支持调用各种返回值类型的实例方法。调用一个定义在父类中的实例方法,须遵循下面的步骤。
  使用 GetMethodID 函数从一个指向父类的 Class 引用当中获取方法 ID。

//4. 获取animal类Class引用
cls_animal = env->FindClass("com/example/gnaix/ndk/Animal");
if(cls_animal == NULL){
    LOGD("cls_dog null");
    return;
}

//5. 获取父类run方法ID并调用
mid_run = env->GetMethodID(cls_animal, "run", "()V");
if(mid_run == NULL){
    LOGD("mid_run null");
    return;
}

  传入子类对象、父类 Class 引用、父类方法 ID 和参数,并调用 CallNonvirtualVoidMethodCallNonvirtualBooleanMethodCallNonvirtualIntMethod 等一系列函数中的一个。其中CallNonvirtualVoidMethod 也可以被用来调用父类的构造函数。

env->CallNonvirtualVoidMethod(obj_dog, cls_animal, mid_run);
点赞
收藏
评论区
推荐文章
刘望舒 刘望舒
4年前
Android深入理解JNI(一)JNI原理与静态、动态注册
Android框架层Android深入理解JNIAndroid框架层本文首发于微信公众号「刘望舒」前言JNI不仅仅在NDK开发中应用,它更是Android系统中Java与Native交互的桥梁,不理解JNI的话,你就只能停留在JavaFramework层。这一个系列我们来一起深入学习JNI。<!more1.JNI概述Android系统按语言来划分的
全靠这份Android知识点PDF大全,月薪30K
第一阶段:Android基础知识回顾:回顾Android开发编程,深入理解Android系统原理和层次结构,深入分析Handler源码和原理;回顾Java,C/C,Kotlin、dart在Android开发中必用的语言,熟悉一下几种语言混淆后的特性;回顾AndroidIPC和JNI的底层原理和热更新技术回顾Native开发要点,使用C结
Stella981 Stella981
3年前
SpringBoot学习:整合shiro自动登录功能(rememberMe记住我功能)
首先在shiro配置类中注入rememberMe管理器!复制代码(https://oscimg.oschina.net/oscnet/675f5689159acfa2c39c91f4df40a00ce0f.gif)/cookie对象;rememberMeCookie()方法是设置Cookie的生成模
Stella981 Stella981
3年前
Android NDK 开发入门环境配置与小例子
NDK:NativeDevelopmentKit原生开发工具NDK能干什么:NDK使得在android中,java可以调用C函数库。为什么要用NDK:我们都知道,java是半解释型语言,很容易被反汇编后拿到源代码文件,在开发一些重要协议时,我们为了安全起见,使用C语言来编写这些重要的部分,来增大系统的安全性。还有,在一些接近硬件环境下,相信大家都
Stella981 Stella981
3年前
NDK集成libjpeg和libpng
最近要在android上使用libjpeg和libpng库来做些图片的处理工作,下载了源码,在pc上使用configure&make&sudomakeinstall,然后参照example.c写了一些例子,都还不错。但是现在要移植到android里面,就需要使用NDK来进行编译了,试了一些交叉编译的方法,由于自己对这方面也不是很了解,所以效果
Wesley13 Wesley13
3年前
JNI实战全面解析
简介项目决定移植一款C开源项目到Android平台,开始对JNI深入研究。JNI是什么?JNI(JavaNativeInterface)意为JAVA本地调用,它允许Java代码和其他语言写的代码进行交互,简单的说,一种在Java虚拟机控制下执行代码的标准机制。NDK是什么?AndroidNDK(NativeDe
Wesley13 Wesley13
3年前
Java开发者容易犯的十个错误
!(https://oscimg.oschina.net/oscnet/c9f00cc918684fbe8a865119d104090b.gif)Top1.数组转换为数组列表将数组转换为数组列表,开发者经常会这样做:\java\List<StringlistArrays.asList(arr);Arr
Easter79 Easter79
3年前
SpringBoot学习:整合shiro自动登录功能(rememberMe记住我功能)
首先在shiro配置类中注入rememberMe管理器!复制代码(https://oscimg.oschina.net/oscnet/675f5689159acfa2c39c91f4df40a00ce0f.gif)/cookie对象;rememberMeCookie()方法是设置Cookie的生成模
Stella981 Stella981
3年前
Android JNI开发系列(七)访问数组
JNI访问数组JNI中的数组分为基本类型数组和对象数组,它们的处理方式是不一样的,基本类型数组中的所有元素都是JNI的基本数据类型,可以直接访问。而对象数组中的所有元素是一个类的实例或其它数组的引用,和字符串操作一样,不能直接访问Java传递给JNI层的数组,必须选择合适的JNI函数来访问和设置Java层的数组对象。
Stella981 Stella981
3年前
Android NativeCrash 捕获与解析
Android开发中,NE一直是不可忽略却又异常难解的一个问题,原因是这里面涉及到了跨端开发和分析,需要同时熟悉Java,C&C,并且需要熟悉NDK开发,并且解决起来不像Java异常那么明了,本文为了解决部分疑惑,将从NE的捕获,解析与还原等三个方面进行探索。一、NE简介NE全称NativeCrash,就是C或者C运
融云IM即时通讯 融云IM即时通讯
8个月前
融云IM干货丨如何确保在项目中只包含一个libc++_shared.so版本?
确保项目中只包含一个libcshared.so版本的关键在于统一C运行时,并合理配置项目的构建脚本。以下是一些具体的步骤和方法:统一NDK版本:确保项目中所有模块使用的NDK版本一致,这有助于避免不同版本NDK生成的libcshared.so之