您当前的位置:首页 > 计算机 > 编程开发 > Java

JNI操作接口实践(三)

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

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

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