前面我们介绍了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时注册方法数组。