美文网首页
android-jni RegisterNatives注册本地方

android-jni RegisterNatives注册本地方

作者: MrOnion0603 | 来源:发表于2017-08-29 17:31 被阅读0次

    RegisterNatives的使用方法

    在Android中通过jni调用本地方法(c/c++),通常使用javah生成规范的头文件,定义一个函数实现本地方法,函数名必须使用本地方法的全类名的规范来写。

    Java_ + 包名 + 类名+ 接口名

    注意名字之间用下划线,下面是示例:
    JNIEXPORT jstring Java_com_example_test_MainActivity_helloworld(JNIEnv *, jclass );

    有没有觉得这种写法太反人类了,其实jni还提供了RegisterNative函数手动的注册native方法,该方法可以自由命名函数,不必像上述方法那样拘泥特定烦杂的命名方式。
    RegisterNatives使用示例:

    static int registerNativeMethods(JNIEnv* env)
    {
        jclass clazz;
        clazz = env->FindClass("com/example/test/MainActivity");
        if (clazz == NULL) {
            return JNI_FALSE;
        }
        if (env->RegisterNatives(clazz, gMethods, sizeof(gMethods) / sizeof(gMethods[0])) < 0) {
            return JNI_FALSE;
        }
    
        return JNI_TRUE;
    }
    

    RegisterNatives中第二个参数gMethods是一个二维数组,代表着这个class里的每一个native方法所对应的实现的方法。写法如下示例:

    static JNINativeMethod gMethods[] = {
        {"helloworld", "()Ljava/lang/String;", (void*)Jni_helloworld}};
    

    第三个参数代表要指定的native的数量。
    这时将前面在jni中声明的

    jstring JNIEXPORT jstring Java_com_example_test_MainActivity_helloworld(JNIEnv *, jclass );
    

    改为

    jstring helloworld(JNIEnv *, jclass);
    

    是不是清爽了很多!

    JNI字段描述符

    在使用RegisterNatives注册本地方法中比较容易出错的地方就是JNINativeMethod gMethods[]其中的()Ljava/lang/String;
    先解释下()Ljava/lang/String;的含义,它是一种对函数返回值和参数的编码,这种编码叫做JNI字段描述符(JavaNative Interface FieldDescriptors)。
    JNI字段描述符的规则是在括号里放置参数,在括号后面放置返回类型:

    (参数描述符)返回类型

    例如()Ljava/lang/String;表示对应的java中的helloworld方法无传入参数,返回类型为String,即String helloworld();
    这个示例可能过于简单,下面再举几个例子:
    (Ljava/lang/String;Ljava/lang/Object;)Ljava/lang/String;表示String f(String a,Object b);
    ([B)Ljava/lang/String;表示String f(byte [] bytes);这里B表示java中的byte类型,[表示数组,[B表示byte []
    以此类推,
    [Ljava/lang/String;表示String []
    [Ljava/lang/Object;表示Object[]
    如果是二维数组int[][],用[[I表示。
    另外要注意的是,引用类型(除基本类型的数组外)通常以"L"开头,以";"结尾,中间是用"/" 隔开的包及类名。所以";"分号是属于引用类型的一部分,因此参数中如果有多个参数的话是不用;间隔的。如:
    (BI)V表示void f(byte b, int i);
    (ILjava/lang/String;)V表示void f(int i,String s);

    最后附上java类型与jni中字符对应的关系表:

    Java 类型 符号
    Boolean Z
    Byte B
    Char C
    Short S
    Int I
    Long J
    Float F
    Double D
    Void V
    objects对象 L开头,以;结尾,中间是用/ 隔开的包及类名。比如:Ljava/lang/String;如果是嵌套类,则用$来表示嵌套。例如 (Ljava/lang/String;Landroid/os/FileUtils$FileStatus;)Z

    相关文章

      网友评论

          本文标题:android-jni RegisterNatives注册本地方

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