美文网首页Android开发常见小问题列表
使用JNI_OnLoad动态注册函数

使用JNI_OnLoad动态注册函数

作者: arkliu | 来源:发表于2018-07-22 18:48 被阅读1032次

    在应用层加载so的时候,虚拟机首先回去/自动执行JNI_OnLoad(...),

    传统java Jni方式:
    1.编写带有native方法的Java类
    2.使用javah命令生成.h头文件
    3.编写代码实现头文件中的方法

    但是上述每次都需要通过javah依据java类的全类名生成对应的native函数全名称,其实我们可以使用RegisterNatives方法把c/c++中的方法隐射到Java中的native方法

    使用JNI_OnLoad步骤

    • 在c/c++文件中定义并实现对应java中声明的本地方法,方法名称随意,但是参数类型和参数个数必须一样

    • 创建声明JNINativeMethod类型的数组,值为需要动态加载映射的本地方法

    • 实现JNI_OnLoad方法,主要分为下面两步:

      • 通过FindClass获取所需的映射的java类
      • 通过jint RegisterNatives(jclass clazz, const JNINativeMethod* methods, jint nMethods) 方法动态注册

    JNI_OnLoad案例

    创建JniOnloadTest.java

    package com.jni.test;
    
    /**
     * Created by liuhang on 18-7-22.
     */
    
    public class JniOnloadTest {
    
        public native int  javaAdd(int x, int y);
        public native String javaSayHi();
    
        static {
            System.loadLibrary("test-lib");
        }
    }
    
    

    创建jniload.cpp

    #include <jni.h>
    #include <string>
    #include <android/log.h>
    #define  LOG_TAG    "liuhang"
    #define  LOGI(...)  __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
    
    static jint JNICALL cAdd(JNIEnv *env, jobject jobj, jint x, jint y){
        LOGI("cAdd x is :%d  y is :%d", x, y);
        return x + y;
    }
    
    static jstring JNICALL cSayHi(JNIEnv *env, jobject jobj, jint x, jint y){
        LOGI("cSayHi runs... will return a string in c");
        return env->NewStringUTF("hello from cSayHi");
    }
    
    /**
       第一个参数:javaAdd 是java中的方法名称
       第二个参数:(II)I  是java中方法的签名,可以通过javap -s -p 类名.class 查看
       第三个参数: (jstring *)cSayHi  (返回值类型)映射到native的方法名称
    
    */
    static const JNINativeMethod gMethods[] = {
            {"javaAdd", "(II)I", (jint *)cAdd},
            {"javaSayHi","()Ljava/lang/String;",(jstring *)cSayHi}
    };
    
    
    static jclass myClass;
    // 这里是java调用C的存在Native方法的类路径
    static const char* const className="com/jni/test/JniOnloadTest";
    JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved){
        LOGI("jni onload called");
        JNIEnv* env = NULL; //注册时在JNIEnv中实现的,所以必须首先获取它
        jint result = -1;
        if(vm->GetEnv((void **) &env, JNI_VERSION_1_4) != JNI_OK) { //从JavaVM获取JNIEnv,一般使用1.4的版本
            return -1;
        }
        // 获取映射的java类
        myClass = env->FindClass(className);
        if(myClass == NULL)
        {
          printf("cannot get class:%s\n", className);
          return -1;
        }
        // 通过RegisterNatives方法动态注册
        if(env->RegisterNatives(myClass, gMethods, sizeof(gMethods)/sizeof(gMethods[0])) < 0)
        {
          printf("register native method failed!\n");
          return -1;
        }
        LOGI("jni onload called end...");
        return JNI_VERSION_1_4; //这里很重要,必须返回版本,否则加载会失败。
    }
    

    java端调用

    JniOnloadTest jniOnloadTest = new JniOnloadTest();
    Log.d("liuhang", "jnionload add result is :"+jniOnloadTest.javaAdd(1, 2)+" sayhi is :"+jniOnloadTest.javaSayHi());
    

    此时执行代码效果如下:


    4.png

    jnidemo代码下载

    相关文章

      网友评论

        本文标题:使用JNI_OnLoad动态注册函数

        本文链接:https://www.haomeiwen.com/subject/kihrmftx.html