2025年3月29日 星期六 甲辰(龙)年 月廿八 设为首页 加入收藏
rss
您当前的位置:首页 > 计算机 > 编程开发 > Java

JNI操作接口实践(三)

时间:08-15来源:作者:点击数:45

前面我们介绍了JNI的常规注册方法(静态注册方法),并对JNI各种典型应用做了一个实践演练。本文我们介绍JNI的另一个注册方法:动态注册方法

一、静态注册

1、注册步骤

1)编写java代码和声明native 方法

2)过javah指令,自动成JNI的头文件(或者按照JNI规范手动自定义)

3)实现jni头文件里面定义的函数

2、注册原理

静态注册的原理是当加载动态库到jvm后,当Native方法第一次执行时会根据其方法名去匹配对应的C语言实现。

3.优点

可以自动生成头文件,模式相对固定。

4、缺点

1)后期类名、文件名改动,头文件所有函数将失效,需要手动改,超级麻烦易出错

2)代码编写不方便,由于JNI层函数的名字必须遵循特定的格式,且名字特别长;

3)会导致程序员的工作量很大,因为必须为所有声明了native函数的java类编写JNI头文件;

4)程序运行效率低,因为初次调用native函数时需要根据根据函数名在JNI层中搜索对应的本地函数,然后建立对应关系,这个过程比较耗时。

二、动态注册

1、注册步骤

1)建立java函数和C函数映射的数组(签名必须一致)

2)通过RegisterNatives注册映射数组

3)重写JNI_OnLoad方法,动态库加载时就会调用JNI_OnLoad,注册映射数组

2、注册原理

动态注册的关键是JNINativeMethod结构体和JNI_OnLoad的实现,JNINativeMethod结构体包含:name-方法名;signature-方法签名(描述返回值和入参);fnPtr-c中实现的函数指针;JNI_OnLoad作用是绑定JNINativeMethod和class直接的关系并返回JNI的版本号。在执行JNI_OnLoad完成注册后,当java代码中执行Native方法时根据调用类可以找对应JNINativeMethod再根据方法名和方法签名可以找到对应的C语言函数指针。

3.优点

首次执行Native方法时即可根据映射关系直接获取对应的C函数指针,无需全局匹配

对C函数的实现类文件命名没有特殊要求,可以在一个文件里完成多个java文件中多个Native方法的实现和注册。增加新的JINI接口只需要增加注册数组成员,比较灵活,且效率高。

三、动态注册最佳实践

1)Java层代码

  • public native String GetstringFromJNI();
  • // Used to load the 'native-lib' library on application startup.
  • static {
  • System.loadLibrary("jni-test");
  • }
  • private ActivityMainBinding binding;
  • @Override
  • protected void onCreate(Bundle savedInstanceState) {
  • super.onCreate(savedInstanceState);
  • binding = ActivityMainBinding.inflate(getLayoutInflater());
  • setContentView(binding.getRoot());
  • // Example of a call to a native method
  • TextView tv = binding.sampleText;
  • tv.setText(GetstringFromJNI());
  • }

Java代码的声明和调用与常规的静态方法时一致的。

2、Jni层代码

  • #include <jni.h>
  • #include <string>
  • extern "C" JNIEXPORT jstring JNICALL
  • GetstringFromJNI(
  • JNIEnv* env,
  • jobject /* this */) {
  • std::string hello = "Hello from jni test2 by dynamic load";
  • return env->NewStringUTF(hello.c_str());
  • }
  • /*
  • 需要注册的函数列表,放在JNINativeMethod 类型的数组中,以后如果需要增加函数,只需在这⾥添加就⾏了
  • 参数:
  • 1.java中⽤native关键字声明的函数名
  • 2.签名(传进来参数类型和返回值类型的说明)
  • 3.C/C++中对应函数的函数名(地址)*/
  • static JNINativeMethod gMethods[] = {
  • { "GetstringFromJNI", "()Ljava/lang/String;", (void*)GetstringFromJNI },//绑定
  • };
  • //此函数通过调⽤RegisterNatives⽅法来注册我们的函数
  • static int registerNativeMethods(JNIEnv* env, const char* className,
  • JNINativeMethod* getMethods, int methodsNum) {
  • jclass clazz;
  • //找到声明native⽅法的类
  • clazz = env->FindClass(className);
  • if (clazz == NULL) {
  • return JNI_FALSE;
  • }
  • //注册函数 参数:java类 所要注册的函数数组 注册函数的个数
  • if (env->RegisterNatives(clazz, getMethods, methodsNum))
  • {
  • return JNI_FALSE;
  • }
  • return JNI_TRUE;
  • }
  • static int registerNatives(JNIEnv* env) {
  • //指定类的路径,通过FindClass ⽅法来找到对应的类
  • const char* className = "com/example/jni_test2/MainActivity";
  • return registerNativeMethods(env, className, gMethods,
  • sizeof(gMethods) / sizeof(gMethods[0]));
  • }
  • //动态注册 java通过loadLibrary 调用so库就会触发该函数加载jni库
  • JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved) {
  • JNIEnv* env = NULL;
  • //获取JNIEnv
  • if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
  • return -1;
  • }
  • //注册函数 registerNatives ->registerNativeMethods ->env->RegisterNatives
  • if (!registerNatives(env)) {
  • return -1;
  • }
  • //返回jni 的版本
  • return JNI_VERSION_1_6;
  • }

以上就是关于jni动态方法的注册,基本思路:定义方法数组---在jni_load时注册方法数组。

方便获取更多学习、工作、生活信息请关注本站微信公众号城东书院 微信服务号城东书院 微信订阅号
推荐内容
相关内容
栏目更新
栏目热门