java调用jni函数示例 NewStringUTF

2024-05-24 11:58

本文主要是介绍java调用jni函数示例 NewStringUTF,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

对第三个参数 jboolean *isCopy说明如下:
当从JNI函数GetStringUTFChars函数中返回得到字符串B时,如果B是原始字符串java.lang.String的一份拷贝,
则isCopy 被赋值为JNI_TRUE。如果B是和原始字符串指向的是JVM中的同一份数据,则isCopy 被赋值为JNI_FALSE。
当isCopy 为JNI_FALSE时,本地代码绝不能修改字符串的内容,否则JVM中的原始字符串也会被修改,这会打破Java语言
中字符串不可变的规则。
通常,我们不必关心JVM是否会返回原始字符串的拷贝,只需要为isCopy传递NULL作为参数 。

                                                                                                       ----     以上内容来自 《JNI编程指南》

一、java->JNI

一、参数String 返回值String

JNIEXPORT jstring JNICALL getString(JNIEnv *evn, jobject jobject1, jstring data) {jboolean isCopy;//evn:JNI 接口指针,data:Java 字符串对象,isCopy:指向布尔值的指针char *startData = (*evn)->GetStringUTFChars(evn, data, &isCopy);char *endData="Hello";int size1=strlen(startData);int size2= strlen(endData);char resultData[256];int index=-1;for(int k=0;k<size1;k++){resultData[++index]=startData[k];}for(int k=0;k<size2;k++){resultData[++index]=endData[k];}jstring  result=(*evn)->NewStringUTF(evn,resultData);//使用了GetStringUTFChars一定要调用ReleaseStringChars函数释放资源//env:JNI 接口指针,data:Java 字符串对象,指向UTF-8 字符串的指针(*evn)->ReleaseStringUTFChars(evn,data,startData);return  result;
}
jni 方法签名: {"getString","(Ljava/lang/String;)Ljava/lang/String;",(void*)getString},java调用 getString("jni-") 返回:jni-hello二.参数输入byte数组,参数输出byte数组。返回void
JNIEXPORT void JNICALL getByteArray(JNIEnv *evn,jobject jobject1,jbyteArray array,jbyteArray outArray){jboolean  isCopy;jint len=(*evn)->GetArrayLength(evn,array);//获取数组长度jbyte *jByteData = (jbyte *)malloc(len * sizeof(jbyte));//参数1:JNI 接口指针,参数2:java数组对象,参数3:起始下标,参数4:要复制的元素个数,参数5:目的缓冲区(*evn)->GetByteArrayRegion(evn,array,0,len,jByteData);//回传给java的数据   evn:JNI 接口指针,array:Java 数组对象, isCopy:指向布尔值的指针。jbyte *jOutArray=(*evn)->GetByteArrayElements(evn,outArray,&isCopy);for(int k=0;k<len;k++){jOutArray[k]=jByteData[k];}//参数1:jni接口指针,参数2:java数组对象,参数3:指向数组元素的指针,参数4:释放模式//参数4说明如下://  0             复制回内容并释放elems 缓冲区//JNI_COMMIT       复制回内容但不释放elems 缓冲区//JNI_ABORT        释放缓冲区但不复制回变化(*evn)->ReleaseByteArrayElements(evn,outArray,jOutArray,0);
}
jni给方法签名 {"getByteArray","([B[B)V",(void*)getByteArray}
java方法:public native void getByteArray(byte[] data,byte[] outData)

三.参数输入byte数组,返回byte数组

JNIEXPORT jbyteArray JNICALL getByteArray2(JNIEnv *env,jobject jobject1,jbyteArray array){jboolean  isCopy;jint len=(*env)->GetArrayLength(env,array);//获取数组长度jbyte *jByteData = (jbyte *)malloc(len * sizeof(jbyte));//参数1:JNI 接口指针,参数2:java数组对象,参数3:起始下标,参数4:要复制的元素个数,参数5:目的缓冲区(*env)->GetByteArrayRegion(env,array,0,len,jByteData);//创建一个byte数组jbyteArray resultArray =(*env)->NewByteArray(env,len);//参数1:JNI 接口指针,参数2:java目标数组对象,参数3:起始下标,参数4:要复制的元素个数,参数5:源缓冲区(*env)->SetByteArrayRegion(env,resultArray,0,len, jByteData);return  resultArray;}

jni签名:{“getByteArray2”,"([B)[B",(void*)getByteArray2}
java定义 :public native byte[] getByteArray2(byte[] data);
二、JNI->java

一、NDK里面获取DeviceId
JNIEXPORT jstring JNICALL getDevice(JNIEnv *env,jobject jobject1,jobject mContext){

 jstring error=(*env)->NewStringUTF(env,"error");
//获取java class 对象  参数1:JNI 接口指针,参数2:java包名+类名jclass  context=(*env)->FindClass(env, "android/content/Context");if(context==NULL){return error;}
//获取类属性   参数1:JNI 接口指针,  参数2:Java 类对象,  参数3: 属性名称; 参数4:属性签名jfieldID  TELEPHONY_SERVICE_ID=(*env)->GetStaticFieldID(env,context,"TELEPHONY_SERVICE","Ljava/lang/String;");if(TELEPHONY_SERVICE_ID==NULL){return  error;}
//获取属性值  参数1:JNI接口指针,    参数2:Java类对象, 参数3:jfieldIDjstring TELEPHONY_SERVICE=((*env)->NewStringUTF(env,"phone"));// (*env)->GetStaticObjectField(env,context,TELEPHONY_SERVICE_ID);//获取java方法  参数1:JNI接口指针,    参数2:Java类对象, 参数3:方法名称, 参数4:参数和返回值签名jmethodID getSystemService=(*env)->GetMethodID(env,context,"getSystemService","(Ljava/lang/String;)Ljava/lang/Object;");if(getSystemService==NULL){return error;}//调用java方法   参数1:JNI接口指针,    参数2:Java类对象, 参数3:方法名称, 参数4:方法参数jobject jobject2=(*env)->CallObjectMethod(env,mContext,getSystemService,TELEPHONY_SERVICE);
if(jobject2==NULL){return error;
}
jclass  TelephonyManager=(*env)->FindClass(env,"android/telephony/TelephonyManager");
if(TelephonyManager==NULL){return error;
}
jmethodID getDeviceID=(*env)->GetMethodID(env,TelephonyManager,"getDeviceId","()Ljava/lang/String;");
if(getDeviceID==NULL){return error;
}
jstring id=(*env)->CallObjectMethod(env,jobject2,getDeviceID);
return id;

}
————————————————
版权声明:本文为CSDN博主「jtzp007」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/JTZP007/article/details/79663284

这篇关于java调用jni函数示例 NewStringUTF的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Boot配置和使用两个数据源的实现步骤

《SpringBoot配置和使用两个数据源的实现步骤》本文详解SpringBoot配置双数据源方法,包含配置文件设置、Bean创建、事务管理器配置及@Qualifier注解使用,强调主数据源标记、代... 目录Spring Boot配置和使用两个数据源技术背景实现步骤1. 配置数据源信息2. 创建数据源Be

Spring Boot 3.x 中 WebClient 示例详解析

《SpringBoot3.x中WebClient示例详解析》SpringBoot3.x中WebClient是响应式HTTP客户端,替代RestTemplate,支持异步非阻塞请求,涵盖GET... 目录Spring Boot 3.x 中 WebClient 全面详解及示例1. WebClient 简介2.

Java中使用 @Builder 注解的简单示例

《Java中使用@Builder注解的简单示例》@Builder简化构建但存在复杂性,需配合其他注解,导致可变性、抽象类型处理难题,链式编程非最佳实践,适合长期对象,避免与@Data混用,改用@G... 目录一、案例二、不足之处大多数同学使用 @Builder 无非就是为了链式编程,然而 @Builder

在IntelliJ IDEA中高效运行与调试Spring Boot项目的实战步骤

《在IntelliJIDEA中高效运行与调试SpringBoot项目的实战步骤》本章详解SpringBoot项目导入IntelliJIDEA的流程,教授运行与调试技巧,包括断点设置与变量查看,奠定... 目录引言:为良驹配上好鞍一、为何选择IntelliJ IDEA?二、实战:导入并运行你的第一个项目步骤1

Spring Boot从main方法到内嵌Tomcat的全过程(自动化流程)

《SpringBoot从main方法到内嵌Tomcat的全过程(自动化流程)》SpringBoot启动始于main方法,创建SpringApplication实例,初始化上下文,准备环境,刷新容器并... 目录1. 入口:main方法2. SpringApplication初始化2.1 构造阶段3. 运行阶

Spring Boot3.0新特性全面解析与应用实战

《SpringBoot3.0新特性全面解析与应用实战》SpringBoot3.0作为Spring生态系统的一个重要里程碑,带来了众多令人兴奋的新特性和改进,本文将深入解析SpringBoot3.0的... 目录核心变化概览Java版本要求提升迁移至Jakarta EE重要新特性详解1. Native Ima

Java中的xxl-job调度器线程池工作机制

《Java中的xxl-job调度器线程池工作机制》xxl-job通过快慢线程池分离短时与长时任务,动态降级超时任务至慢池,结合异步触发和资源隔离机制,提升高频调度的性能与稳定性,支撑高并发场景下的可靠... 目录⚙️ 一、调度器线程池的核心设计 二、线程池的工作流程 三、线程池配置参数与优化 四、总结:线程

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三

我们来说说Java LockSupport 的 park 和 unpark

《我们来说说JavaLockSupport的park和unpark》LockSupport是JDK底层线程阻塞工具,通过park/unpark实现线程阻塞与唤醒,避免死锁,与Object的w... 目录一、LockSupport1.1、LockSupport函数列表1.2、基本使用先 park 再 unpa

SpringBoot集成MyBatis实现SQL拦截器的实战指南

《SpringBoot集成MyBatis实现SQL拦截器的实战指南》这篇文章主要为大家详细介绍了SpringBoot集成MyBatis实现SQL拦截器的相关知识,文中的示例代码讲解详细,有需要的小伙伴... 目录一、为什么需要SQL拦截器?二、MyBATis拦截器基础2.1 核心接口:Interceptor