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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听