当前位置: 首页 > 编程日记 > 正文

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

上一篇有提到 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和方法IDfid = 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);
}

运行结果

代码解析

第一步:调用 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. 获取类的默认构造函数IDmid_construct = env->GetMethodID(clazz, "<init>", "()V");if(mid_construct == NULL){LOGD("construct null");return;}//3. 获取实例方法ID和变量IDfid_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);
}

运行结果

代码解析

访问实例变量和方法没有很大区别多了一个实例化的步骤。每个类会生成一个默认的构造函数,调用的时候用 <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...");}@Overridepublic String getName() {return "My name is " + this.name;}@Overridepublic 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类构造方法IDmid_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);
}

运行结果

代码解析

调用构造方法

调用构造方法和调用对象的实例方法方式是相似的,传入”< 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);

相关文章:

在C#中应用哈希表(Hashtable)

一,哈希表(Hashtable)简述 在.NET Framework中&#xff0c;Hashtable是System.Collections命名空间提供的一个容器&#xff0c;用于处理和表现类似key/value的键值对&#xff0c;其中key通常可用来快速查找&#xff0c;同时key是区分大小写&#xff1b;value用于存储对应于key的…

俄罗斯自研Elbrus CPU参数曝光,CEO年近九旬仍未退休

导语&#xff1a;俄罗斯自研 CPU 参数最近曝光&#xff0c;虽然比起主流产品仍存在较大差距&#xff0c;但是这也是俄罗斯在自研道路上的一大进展。虽说自研 CPU 并非易事&#xff0c;但为了避免被美国牵制&#xff0c;很多国家都在这方面投入了巨大的人力与资金。战斗民族俄罗…

停止Password Manager Agent服务导致应用程序启动缓慢

在一个实施环境中&#xff0c;部署了Password Manager用来实现单点登录功能&#xff0c;但是由于Password Manager的提示基本都是以英文为主&#xff0c;而且配置也比较麻烦&#xff0c;普通用户看见会比较影响用户体验&#xff0c;所以用户决定暂时关闭Password Manager功能&a…

web 前端常用组件【06】Upload 控件

因为有万恶的IE存在&#xff0c;所以当Web项目初始化并进入开发阶段时。 如果是项目经理&#xff0c;需要知道客户将会用什么浏览器来访问系统。 明确知道限定浏览器的情况下&#xff0c;你才能从容的让手下的封装必要的前端组件。 本篇文章试图从常见的上传方式和组件进行分析…

性能超越最新序列推荐模型,华为诺亚方舟提出记忆增强的图神经网络

作者 | Chen Ma, Liheng Ma等译者 | Rachel出品 | AI科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09;用户-商品交互的时间顺序可以揭示出推荐系统中用户行为随时间演进的序列性特征。用户与之交互的商品可能受到用户曾经接触的商品的影响。但是&#xff0c;用户和商…

ASP.net 中的页面继承实现和通用页面的工厂模式的实现

最近用.Net做web项目的时候遇到了一些问题&#xff0c;就是很多的页面的处理一样的&#xff0c;不一样的就是我们写的存储过程不同&#xff0c;为了考虑代码的重复利用和可维护性和可 扩展性&#xff0c;于是写了一个对于单据页面的工厂模式&#xff0c;采用界面的继承技术&…

5502的时钟组

5502有四个时钟组&#xff0c;分别为&#xff1a; C55x Subsystem Clock GroupFast Peripherals Clock GroupSlow Peripherals Clock GroupExternal Memory Interface Clock Group1、C55x Subsystem Clock Group该时钟组包括C55X CPU core、内存&#xff08;DARAM和ROM&#xf…

遇到的浏览器兼容问题及应对方法

前言&#xff1a; 上周天的时候有个学长找我帮忙做三张页面&#xff0c;因为没有数据交换之类的&#xff0c;只是单纯的前端页面&#xff0c;想着好久没做东西&#xff0c; 看书都看烦了&#xff0c;所以就接了也当是练手。之前因为没有系统的看书&#xff0c;所以其实很多问题…

浅谈权限设计(来自深空老大)

2019独角兽企业重金招聘Python工程师标准>>> By 深空, 2009-09-13 21:45:07 PHPChina的专家版在谈权限设计&#xff0c;苦于没有权限回帖&#xff0c;特发此博文谈谈简单的权限设计。讨论在这里。 最简单的权限验证&#xff0c;应该是登录态的验证&#xff0c;如果登…

5年Python功力,总结了10个开发技巧

作者 | 写代码的明哥来源 |Python编程时光&#xff08;ID: Cool-Python&#xff09;如何在运行状态查看源代码&#xff1f;查看函数的源代码&#xff0c;我们通常会使用 IDE 来完成。比如在 PyCharm 中&#xff0c;你可以 Ctrl 鼠标点击 进入函数的源代码。那如果没有 IDE 呢&…

怎样给目录加权限0777

# chmod -R 0777 /var/www/html/子目录

php学习,一个简单的Calendar(2) 一个简单的活动页面

有了前面的基础&#xff0c;后面就是将页面展示出来。 预览图如下&#xff1a;1号和31号分别有活动&#xff0c;会一并显示出来 这里需要搞定几个问题&#xff0c;一个就是数据库的连接&#xff0c;我们用\sys\class\class.db_connect.inc.php <?php /* * 数据库操作&#…

涨见识了,在终端执行 Python 代码的 6 种方式

作者 | BRETT CANNON译者 | 豌豆花下猫Python猫为了我们推出的 VS Code 的 Python 插件[1]&#xff0c;我写了一个简单的脚本来生成变更日志[2]&#xff08;类似于Towncrier[3]&#xff0c;但简单些&#xff0c;支持 Markdown&#xff0c;符合我们的需求&#xff09;。在发布过…

ASP.NET中DataGrid鼠标经过感知以及点击行弹出窗口

选择自 xujh 的 Blog 作者Blog&#xff1a;http://blog.csdn.net/xujh/ 很多人说很难&#xff0c;其实就这几行代码。只要在DataGrid1的ItemDataBound中写入下代码即可 private void DataGrid1_ItemDataBound(object sender, System.Web.UI.WebControls.DataGridItemEventAr…

python中的module

Python中的Module是比较重要的概念。常见的情况是&#xff0c;事先写好一个.py文件&#xff0c;在另一个文件中需要import时&#xff0c;将事先写好的.py文件拷贝到当前目录&#xff0c;或者是在sys.path中增加事先写好的.py文件所在的目录&#xff0c;然后import。这样的做法&…

找子串替换(kmp)poj1572

题目链接&#xff1a;http://poj.org/problem?id1572 输入数据时要注意&#xff0c;这里是string型 用getline(cin,origin[i]); #include <string> #include <iostream> #include <algorithm> #include <stdio.h>using namespace std;const int maxn …

dll的概念、dll导出类(转)

1、 DLL的概念DLL(Dynamic Linkable Library)&#xff0c;动态链接库&#xff0c;可以向程序提供一些函数、变量或类。这些可以直接拿来使用。静态链接库与动态链接库的区别&#xff1a;&#xff08;1&#xff09;静态链接库与动态链接库都是共享代码的方式。静态链接库把最后的…

墨奇科技汤林鹏:如何用 AI 技术颠覆指纹识别?

受访者 | 墨奇科技联合创始人& CTO 汤林鹏 记者 | Aholiab&#xff0c;编辑 | Carol 出品 | AI科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09; 随着深度学习等AI技术的成熟&#xff0c;生物识别成为了关注度较高的领域&#xff0c;指纹、人脸、虹膜等识别技术…

ASP.Net ViewState的实现

选择自 timmy3310 的 Blog ViewState是.Net中提出的状态保存的一种新途径&#xff08;实际上也是老瓶装新酒&#xff09;&#xff1b;我们知道&#xff0c;传统的Web程序保存状态的方式有这样几种&#xff1a;1、Application 这是Web应用程序生命期中的全局保存区&#xff0c…

【51CTO学院三周年】遇到

作为一名二流学校的大学生&#xff0c;因为学校的一门嵌入式Linux应用程序开发而喜欢上了嵌入式&#xff0c;但是学校却是只上了一学期的课&#xff0c;无奈只能自己找教程继续学习。在3个月前&#xff0c;无意中找到了朱有鹏老师的嵌入式学习基础视频&#xff0c;通过老师视频…

ASP.NET图象处理详解

作者&#xff1a;未知 请与本人联系在使用ASP的时候&#xff0c;我们时常要借助第三方控件来实现一些图象功能。而现在&#xff0c;ASP.NET的推出&#xff0c;我们已经没有必要再使用第三方控件来实现&#xff0c;因为ASP.NET 已经具有强大的功能来实现一些图象处理。现在&…

IT工作者,你们的爱情是这样的吗?

今天在博客里看到了这篇文章&#xff0c;看完这个视频我随笔写了点自己的感受和看法&#xff0c; 视频链接在下方&#xff1a; http://leidu.blog.51cto.com/3245712/622534 很感谢90 男孩提供&#xff0c;建议IT人员看一下&#xff0c;看完写一下你们的感受吧&#xff01;…

平头哥玄铁处理器Linux新版本,5大亮点速览

来源 | 芯片开放社区为了便于 CPU 评估&#xff0c;系统集成&#xff0c;快速上手玄铁处理器 Linux 操作系统&#xff0c;平头哥更新了玄铁处理器 linux 版本&#xff0c;结合 gitlab 开源 CI/CD 系统&#xff0c;对已发布到开源社区的玄铁架构 CPU 相关的生态软件形成持续保障…

费用保险单,如何失焦时自动补零

费用&#xff0c;如何失焦时自动补零转载于:https://www.cnblogs.com/maojiayan/p/5606247.html

腾讯AI种番茄双丰收:参赛AI全胜专家,辽宁试点净利增千元

6月9日&#xff0c;腾讯宣布了两项AI农业领域进展。在研究侧&#xff0c;腾讯 AI Lab 与荷兰瓦赫宁根大学&#xff08;下称WUR&#xff09;联办的“第二届国际智慧温室种植挑战赛”&#xff08;下称比赛&#xff09;落幕。在全球疫情肆虐之时&#xff0c;复赛的五支队伍挑战用 …

在ASP.NET中值得注意的两个地方

在ASP.NET中ASPX页面的Page_Load事件有两个让人奇怪的地方&#xff0c;你应该记住它们&#xff1a; a.有时Page_Load事件在你的ASP.NET页面里会发生多次。这种情况发生的一个可能的原因是你把ASPX页面的AutoEvenWireup值设置成了True。如果是这样&#xff0c;那么在“Sub Page…

yii 级联删除

alter table 外键表 add constraint 级联删除名 foreign key (外键字段) references 主表名(主表字段)ON delete CASCADE go 转载于:https://blog.51cto.com/dasangshu/624605

iOS--优秀博客记录

感谢唐巧整理&#xff0c;我又加了一些备注。原地址&#xff1a;https://github.com/tangqiaoboy/iOSBlogCN 博客地址RSS地址psOneVs Denhttp://onevcat.com/atom.xml 一只魔法师的工坊http://blog.ibireme.com/feed/图片处理、YYKit破船之家http://beyondvincent.com/atom.xml…

linux环境下和网络服务相关的配置文件含义及如何配置

要建立一个安全Linux服务器就首先要了解Linux环境下和网络服务相关的配置文件的含义及如何进行安全的配置。那天查看服务器的eth0地址&#xff0c;后来想了一些问题&#xff0c;到家里就翻了翻以前的文档&#xff0c;无意中看到了这个&#xff0c;现在发布出来&#xff0c;希望…

ASP.NET保持用户状态的九种选择

2003-06-10 ■陶刚编译 ■yesky摘要&#xff1a;ASP.NET为保持用户请求之间的数据提供了多种不同的途径。你可以使用Application对象、cookie、hidden fields、Sessions或Cache对象&#xff0c;以及它们的大量的方法。决定什么时候使用它们有时很困难。本文将介绍了上述的技术&…