Android JNI 访问Java中的成员变量及非native方法

2024-09-05 07:58

本文主要是介绍Android JNI 访问Java中的成员变量及非native方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JNI的头文件生成及编译请参考《Android中的JNI使用指南一》

1、修改Java类中的成员变量。通过HelloJNI.java类中changeNum()本地方法来修改成员number的值。number初始化为0,通过本地方法changeNum()修改为100。HelloJNI.java类中定义如下:

changeNum()在C++中的实现,如下:

//修改java类中的成员变量(int类型),即java在C++中的回调
JNIEXPORT void JNICALL Java_com_example_hellojni_HelloJNI_changeNum(JNIEnv *env, jobject obj)
{jclass cls = env->GetObjectClass(obj);jfieldID id_number = env->GetFieldID(cls, "number", "I");if (id_number == NULL){return;}jint number = env->GetIntField(obj, id_number);LOGD("c.height = %d", number);env->SetIntField(obj, id_number, 100L);
}

在Activity中调用如下:

//方法回调,修改java中成员变量HelloJNI helloJNI = new HelloJNI();helloJNI.changeNum();textView.setText("" + helloJNI.number); //将C++中修改的number值打印出来

2、 修改JAVA类中的静态成员变量,通过HelloJNI.java类中的本地staticAccessField方法来修改静态成员si的值,如图

 C++中的实现如下:
 

//访问静态字段
JNIEXPORT void JNICALL Java_com_example_hellojni_HelloJNI_staticAccessField(JNIEnv *env, jobject obj)
{jfieldID fid; //store the field ID;jint si;jclass cls = env->GetObjectClass(obj);fid = env->GetStaticFieldID(cls, "si", "I");if (fid == NULL){return;}si = env->GetStaticIntField(cls, fid);LOGD("si = %d", si);env->SetStaticIntField(cls, fid, 200);
}

 在Activity中调用如下:

HelloJNI.si = 100;
helloJNI.staticAccessField();
textView.setText("" + HelloJNI.si); //将c++中改变的si值打印出来

3、访问JAVA类中非静态成员方法。通过HelloJNI.java类中的本地nativeMethod方法来访问HelloBean.java类中的test跟add方法。

HelloBean.java类的定义:

package com.example.hellojni;import android.util.Log;public class HelloBean {public static final String TAG = "JNI_";public static void gunder(){//System.out.println("gunder_In Java.");Log.d(TAG, "gunder_ static In Java.");}public  void test(){Log.d(TAG, "gunder_ In Java.");}public int add(int number){Log.d("JNI_", "gunder_ In Java. num is " + number);return number + 2;}}

C++中的实现:

//调用java中的方法
JNIEXPORT void JNICALL Java_com_example_hellojni_HelloJNI_nativeMethod(JNIEnv *env, jobject obj)
{jint num = 20;LOGD("In C++ begin");jclass clazz = NULL;jobject jobj = NULL;jmethodID mid_construct = NULL;jmethodID mid_instance = NULL;jmethodID mid_instance2 = NULL;//1.从classpath路径下搜索ClassMethod这个类,并返回该类的Class对象clazz =  env->FindClass("com/example/hellojni/HelloBean");if (clazz == NULL){LOGD("No find com/example/hellojni/HelloBean class");return;}//2.获取类的默认构造方法IDmid_construct = env->GetMethodID(clazz, "<init>", "()V");if (mid_construct == NULL){LOGD("No find defualt construct.");return ;}//3.查找实例方法的IDmid_instance = env->GetMethodID(clazz, "test","()V");if (mid_instance == NULL){LOGD("No find test method");return;}mid_instance2 = env->GetMethodID(clazz, "add","(I)I");if (mid_instance == NULL){LOGD("No find add method");return;}//4.创建该类的实例jobj = env->NewObject(clazz, mid_construct);if (jobj == NULL){LOGD("No find test in the com/example/hellojni/HelloBean");}//5.调用对象的实例方法env->CallVoidMethod(jobj, mid_instance); //不带参数的方法env->CallIntMethod(jobj, mid_instance2, num); //传递参数的方法//6.删除局部引用env->DeleteLocalRef(clazz);env->DeleteLocalRef(jobj);LOGD("In C++ end");
}

 在Activity中调用如下:

HelloJNI helloJNI = new HelloJNI();
helloJNI.nativeMethod();

 4、访问JAVA类中静态成员方法。通过HelloJNI.java类中的本地nativeStaticMethod方法来访问HelloBean.java类中的gunder方法。如下:

C++中的实现:
 

//调用java中的静态方法
JNIEXPORT void JNICALL Java_com_example_hellojni_HelloJNI_nativeStaticMethod(JNIEnv *env, jclass cls)
{LOGD("In C++ begin static");jclass classTmp =  env->FindClass("com/example/hellojni/HelloBean");if (classTmp == NULL){LOGD("In C++classTmp == NULL");return;}//jclass clazzTts = env->NewGlobalRef(classTmp);
//	if (clazzTts != NULL){jmethodID gunder  = env->GetStaticMethodID(classTmp,"gunder","()V");if (gunder != NULL)env->CallStaticVoidMethod(classTmp,gunder);else{LOGD("stop--- method id error.");}//}LOGD("In C++ end static");
}

  在Activity中调用如下:

HelloJNI.nativeStaticMethod();

代码参考: https://github.com/gunder1129/android-tool/tree/master/HelloJNI

 

 

 

 

这篇关于Android JNI 访问Java中的成员变量及非native方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138383

相关文章

Java中Arrays类和Collections类常用方法示例详解

《Java中Arrays类和Collections类常用方法示例详解》本文总结了Java中Arrays和Collections类的常用方法,涵盖数组填充、排序、搜索、复制、列表转换等操作,帮助开发者高... 目录Arrays.fill()相关用法Arrays.toString()Arrays.sort()A

Spring Boot Maven 插件如何构建可执行 JAR 的核心配置

《SpringBootMaven插件如何构建可执行JAR的核心配置》SpringBoot核心Maven插件,用于生成可执行JAR/WAR,内置服务器简化部署,支持热部署、多环境配置及依赖管理... 目录前言一、插件的核心功能与目标1.1 插件的定位1.2 插件的 Goals(目标)1.3 插件定位1.4 核

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

使用zip4j实现Java中的ZIP文件加密压缩的操作方法

《使用zip4j实现Java中的ZIP文件加密压缩的操作方法》本文介绍如何通过Maven集成zip4j1.3.2库创建带密码保护的ZIP文件,涵盖依赖配置、代码示例及加密原理,确保数据安全性,感兴趣的... 目录1. zip4j库介绍和版本1.1 zip4j库概述1.2 zip4j的版本演变1.3 zip4

Java堆转储文件之1.6G大文件处理完整指南

《Java堆转储文件之1.6G大文件处理完整指南》堆转储文件是优化、分析内存消耗的重要工具,:本文主要介绍Java堆转储文件之1.6G大文件处理的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言文件为什么这么大?如何处理这个文件?分析文件内容(推荐)删除文件(如果不需要)查看错误来源如何避

SpringBoot整合Dubbo+ZK注册失败的坑及解决

《SpringBoot整合Dubbo+ZK注册失败的坑及解决》使用Dubbo框架时,需在公共pom添加依赖,启动类加@EnableDubbo,实现类用@DubboService替代@Service,配... 目录1.先看下公共的pom(maven创建的pom工程)2.启动类上加@EnableDubbo3.实

SpringBoot整合(ES)ElasticSearch7.8实践

《SpringBoot整合(ES)ElasticSearch7.8实践》本文详细介绍了SpringBoot整合ElasticSearch7.8的教程,涵盖依赖添加、客户端初始化、索引创建与获取、批量插... 目录SpringBoot整合ElasticSearch7.8添加依赖初始化创建SpringBoot项

JAVA覆盖和重写的区别及说明

《JAVA覆盖和重写的区别及说明》非静态方法的覆盖即重写,具有多态性;静态方法无法被覆盖,但可被重写(仅通过类名调用),二者区别在于绑定时机与引用类型关联性... 目录Java覆盖和重写的区别经常听到两种话认真读完上面两份代码JAVA覆盖和重写的区别经常听到两种话1.覆盖=重写。2.静态方法可andro

Nginx安全防护的多种方法

《Nginx安全防护的多种方法》在生产环境中,需要隐藏Nginx的版本号,以避免泄漏Nginx的版本,使攻击者不能针对特定版本进行攻击,下面就来介绍一下Nginx安全防护的方法,感兴趣的可以了解一下... 目录核心安全配置1.编译安装 Nginx2.隐藏版本号3.限制危险请求方法4.请求限制(CC攻击防御)

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种