JNI访问属性和方法

简介: NDK连载系列

导读

《JNI之动态注册与静态注册》 一文中,我们介绍了JNI函数动态注册的同时也介绍了JNI中的属性描述符和函数描述符。对这两个概念还不熟悉的童鞋们需要再温习一下。
在今天的文章中我们主要介绍在JNI中如何访问java层对象的属性以及调用java层对象的成员方法等相关知识。

访问java成员属性

1、访问普通属性
在JNI访问java类属性分为两个步骤,首先是通过FindClass函数找到对应的类,然后通过GetFieldID找到对应的属性,如果需要修改变量的话则通过一系列的SetTypeField函数进行修改即可.

下面的例子通过JNI的方式访问了java类Person的name属性并作出修改:

Person.java

public class Person {
    private static int defaultAge = 18;
    private int age;
    private String name;
    Person(int age,String name){
        this.age = age;
        this.name = name;
    }
    public void printName(){
        Log.v("PersonTag","my name is:" + name);
    }

    public static void printDefaultAge(){
        Log.v("PersonTag","default age is:" + defaultAge);
    }

    // 访问名字属性
    public native void changeName();
    // 访问静态属性
    public native void accessStatic();
    // 通过JNI调用java的方法
    public native void printNameByJNI();
    // 访问静态方法
    public native static void callStaticFunc();
    // JNI调用构造方法,生成对象
    public native static Person createPersonByJNI();
}
MainActivity.java

public class MainActivity extends AppCompatActivity {

    static {
        System.loadLibrary("jnitest");
    }

    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = ActivityMainBinding.inflate(getLayoutInflater());
        Person person = new Person(20,"jack");
        person.changeName();
        person.printName();
    }

}
native-lib.cpp

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_Person_changeName(JNIEnv *env, jobject thiz) {
    // 获取对应的类
    jclass jclazz = env->FindClass("com/fly/jnitest/Person");
    // 获取属性,第三个参数是属性描述符
    jfieldID nameFielid = env->GetFieldID(jclazz,"name","Ljava/lang/String;");
    // 修改成员属性
    env->SetObjectField(thiz,nameFielid,env->NewStringUTF("james"));
}

2、访问静态属性
访问静态属性和访问普通属性的流程是一样的,只不过获取的静态属性的函数变成了GetStaticFieldID,还是以上面的代码作为例子,我们通过JNI的方式访问Person类的静态属性defaultAge并作出修改:

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_Person_accessStatic(JNIEnv *env, jobject thiz) {
    // 获取对应的类
    jclass jclazz = env->FindClass("com/fly/jnitest/Person");
    // 获取静态属性,第三个参数是属性描述符
    jfieldID defaultAgeFielid = env->GetStaticFieldID(jclazz,"defaultAge", "I");
    // 修改静态属性
    env->SetStaticIntField(jclazz,defaultAgeFielid,19);
}

调用java成员方法

1、调用普通方法
同访问类的成员属性一样,在JNI中调用java层类的成员方法时也需要先找到对应的类,然后通过JNI函数GetMethodID获取到对应的方法id,就可以使用函数CallVoidMethodCallObjectMethod等完成调用了。

继续是以上面的代码为例,通过JNI的方式,调用Person类对象的printName方法:

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_Person_printNameByJNI(JNIEnv *env, jobject thiz) {
    // 获取对应的类
    jclass jclazz = env->FindClass("com/fly/jnitest/Person");
    // 获取方法id,第三个参数是函数签名
    jmethodID methodId = env->GetMethodID(jclazz,"printName", "()V");
    // 调用java方法
    env->CallVoidMethod(thiz,methodId);
}

2、调用静态方法
调用静态方法与调用普通方法步骤一致,只不过获取方法id的函数换成了GetStaticMethodID,调用的函数换成CallStaticVoidMethod即可,这里就不贴代码了。

3、调用构造函数

调用构造函数和调用普通成员方法一样,也是先获取到对应的类,然后获取到构造函数的jmethodID,最后通过NewObject即可生成java对象,但在获取构造函数的jmethodID时,构造函数的方法名固定为<init>
以下例子展示了在JNI层创建一个Person类并返回给java层:

extern "C"
JNIEXPORT jobject JNICALL
Java_com_fly_jnitest_Person_createPersonByJNI(JNIEnv *env, jclass clazz) {
    // 获取对应的类
    jclass personClazz = env->FindClass("com/fly/jnitest/Person");
    // 获取构造方法,构造方法的函数名称固定为<init>
    jmethodID methodId = env->GetMethodID(personClazz,"<init>", "(ILjava/lang/String;)V");
    // 通过构造方法生成对象
    jobject person = env->NewObject(personClazz,methodId,20,env->NewStringUTF("james"));
    return person;
}

系列推荐

JNI基础简介
JNI之数组与字符串的使用
JNI之动态注册与静态注册

目录
相关文章
|
2月前
|
设计模式
在静态方法中访问类的实例属性和方法时会发生什么?
总之,静态方法主要用于处理与类本身相关的操作和逻辑,不应该直接访问类的实例属性和方法。如果需要在静态方法中使用与实例相关的信息,应该通过合理的参数传递或其他设计模式来实现,以保持代码的清晰性和面向对象设计的原则。
71 8
|
3月前
|
Java
Java“非静态变量 ... 不能在静态上下文中被引用”解决
Java中遇到“非静态变量不能在静态上下文中被引用”的错误,通常是因为尝试在静态方法或静态块中访问实例变量。解决方法是将变量声明为静态(static)或通过实例对象来访问该变量。
375 6
|
3月前
|
Java
Java“非静态方法 ... 不能在静态上下文中被引用”解决
在Java中,“非静态方法……不能在静态上下文中被引用”的错误通常源于在静态方法中直接调用非静态方法。解决方法包括:1) 创建类的实例后调用;2) 将非静态方法改为静态方法;3) 重新评估和调整类的设计以避免此类问题。
614 1
|
8月前
|
存储 Java
Java静态变量在静态方法内部无法改变值
在Java中,静态变量属于类本身,而非类的实例,因此可以在没有创建实例的情况下通过类名访问和修改。若在静态方法中无法改变静态变量的值,可能是因为逻辑错误、局部变量覆盖、误用实例访问或尝试修改`final`静态变量。要访问静态变量,直接通过类名调用即可。修改静态变量同样直接,只需通过类名加变量名并赋新值。静态变量与实例变量的主要区别在于生命周期、存储位置、访问方式和数据共享。静态变量在整个程序运行期间存在,所有实例共享,而实例变量每个对象独有。
|
8月前
|
编译器 C++
C++ 解引用与函数基础:内存地址、调用方法及声明
C++ 中的解引用允许通过指针访问变量值。使用 `*` 运算符可解引用指针并修改原始变量。注意确保指针有效且不为空,以防止程序崩溃。函数是封装代码的单元,用于执行特定任务。理解函数的声明、定义、参数和返回值是关键。函数重载允许同一名称但不同参数列表的函数存在。关注公众号 `Let us Coding` 获取更多内容。
208 1
|
缓存 安全 Java
JNI之缓存与引用
NDK连载系列
79 0
JNI中访问JList的代码
JNI中访问JList的代码
68 0
引用调用
引用调用
119 0
|
架构师 Java
[Java Base] 类,接口,枚举,静态常量到底应该放在哪?
静态常量什么时候适合放在类中?什么时候适合放在接口中?什么时候适合放在枚举中呢?放在不同的type中会引发什么不良后果呢?本篇就来解析一下~常量类应该是final,不变的,而接口里的参数是final,也是不变的。那么,看起来接口是放常量没有一定问题,还省去了final的输入,非常的合适。但是,类是只能单继承的,接口是允许多实现的。要是类实现的多个接口出现重名的常量,会报错,必须要在实现类明确常量用的是哪个接口的。
280 0
|
Java Android开发 C++
【Android NDK 开发】JNI 方法解析 ( C/C++ 调用 Java 方法 | 函数签名 | 调用对象方法 | 调用静态方法 )(一)
【Android NDK 开发】JNI 方法解析 ( C/C++ 调用 Java 方法 | 函数签名 | 调用对象方法 | 调用静态方法 )(一)
555 0
【Android NDK 开发】JNI 方法解析 ( C/C++ 调用 Java 方法 | 函数签名 | 调用对象方法 | 调用静态方法 )(一)