JNI动态注册

Android  NDK  2023年6月28日 am8:08发布1年前 (2023)更新 91es.com站长
52 0 0

前言

之前其实写过,代码不见了,为了走一下流程,重新简单的写了一个。

正文

动态注册

动态注册java的Native方法,使得c/c++方法名可以和java的Native方法名可以不同。动态注册是将二者方法名关联起来,以后在修改Native方法名时,只需修改动态注册关联的方法名称即可。

  1. 优点: 灵活性高, 更改类名,包名或方法时, 只需对更改模块进行少量修改, 效率高

  2. 缺点: 对新手来说稍微有点难理解, 同时会由于搞错签名, 方法, 导致注册失败

实战

定义Hello.java
package com.biumall.dynamic.one;
public class Hello {
    // 1. load Hello.so
    static {
        System.loadLibrary("Hello");
    }
    // 2. define native hello()
    public static native String hello();
}

动态注册需要Java中native的Method(本地方法名)和Signature(签名)

对于Method(本地方法名),很简单就是上面hello。

当对于Signature(签名),需要通过一定的方式获取。

PS:自定义对象不行,需要自己看规则表《JNI之类型介绍

两种方式多需要先javac

#当前位置:BiuJniDynamic\src\main\java\com\biumall\dynamic\one
javac Hello.java

然后在进一步的获取需要的方法和签名

获取签名方式一

在javac基础上进行javap -s

#当前位置:BiuJniDynamic\src\main\java\com\biumall\dynamic\one
javap -s hello.class

命令行就会显示hello()方法的签名。

public static native java.lang.String hello();
  Signature: ()Ljava/lang/String;
获取签名方式二

方式跟静态注册一样,javac后,退回java所在目录,进行javah

#当前位置:BiuJniDynamic\src\main\java
javah com.biumall.dynamic.one.Hello

得到

com_biumall_dynamic_one_Hello.h

.h里面的方法上会有Method和Signature注释

/*
 * Class:     com_biumall_dynamic_one_Hello
 * Method:    hello
 * Signature: ()Ljava/lang/String;
 */

Hello.c

动态注册跟静态不一

系统初始化JNI在加载时,会调用JNI_OnLoad(),而卸载时会调用JNI_UnLoad();所以,我们可以通过重写JNI_OnLoad(),在JNI_OnLoad()中将函数注册到Android中,以便能通过Java访问。

下面是完整代码

#include <jni.h>
#include <android/log.h>
#include<stdio.h>

//LOG_TAG
#define LOG_TAG "from_dynamic_jni_"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)

//获取数组大小
#define ARRAY_LENGTH(x) ((int)(sizeof(x) / sizeof((x)[0])))
//定义Hello.java类路径[包名+类名,只不过.换成了/]
//com.biumall.dynamic.one.Hello
#define DYNAMIC_CLASS "com/biumall/dynamic/one/Hello"

//native 方法
JNIEXPORT jstring JNICALL native_hello
  (JNIEnv * env , jclass jclazz){
    LOGE("native_hello()");
    return (*env)->NewStringUTF(env, "Hello World !!!! --- from JNI ");
};

//定义Java和JNI函数的绑定表
// 方法数组,分别为:(方法名[java层定义的],方法签名,函数指针[c层对应替换的方法])
// 可以通过javac和javap 获取 方法签名
JNINativeMethod method_table[]= {
        {"hello", "()Ljava/lang/String;", (void*) native_hello},
};

int registerNativeMethods(JNIEnv *env, const char *className, JNINativeMethod * methods, int methods_size){
    jclass clazz = NULL;
    //反射Java类
    clazz = (*env)->FindClass(env, className);
    if(NULL == clazz){
        return JNI_ERR;
    }
    int result = (*env)->RegisterNatives(env, clazz, methods, methods_size);
    LOGD("--------registerNativeMethods---------- result : %d", result);
    if(result < 0){
        return JNI_ERR;
    }
    return JNI_OK;
}

JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved){
    JNIEnv * env = NULL;
    int result = (*vm)->GetEnv(vm, (void **) &env, JNI_VERSION_1_6);
    LOGD("--------JNI_OnLoad----------1 result : %d", result);
    if(result != JNI_OK){
    return JNI_ERR;
    }
    result = registerNativeMethods(env , DYNAMIC_CLASS, method_table, ARRAY_LENGTH(method_table));
    LOGD("--------JNI_OnLoad----------2 result : %d", result);
    if(result != JNI_OK){
    return JNI_ERR;
    }
    return JNI_VERSION_1_6;
}

上面大部分有注释,这里就不重复了。

至于其他的Application.mk,Android.mk,ndk-build和build.gradle配置跟静态注册一样,这里就懒得重复了。

可以看最近整理的《JNI静态注册

参考文章

  1. [NDK开发]Android JNI开发之动态注册

  2. [NDK开发]Android JNI 中新增JNI层日志打印

  3. [NDK开发]Android JNI 开发之静态注册

  4. JNI静态注册

 历史上的今天

  1. 2024: 丰子恺:人生三十即是秋(0条评论)
  2. 2020: Glide V4和V3 使用不同(0条评论)
  3. 2018: PhoneStatusBar启动分析(0条评论)
版权声明 1、 本站名称: 91易搜
2、 本站网址: 91es.com3xcn.com
3、 本站内容: 部分来源于网络,仅供学习和参考,若侵权请留言
3、 本站申明: 个人流水账日记,内容并不保证有效

暂无评论

暂无评论...

随机推荐

白岩松:幸福在哪里

一走在人群中,我习惯看一看周围人的手腕,那里似乎藏着一个属于当代中国人的内心秘密,从不言说,却日益增多。越来越多的人,不分男女,会戴上一个手串,这其中,不乏有人仅仅是为了装饰;更多的却带有祈福与安心的意味,这手串停留在装饰与信仰之间,或左或右。这其中,是怎样的一种相信或怎样的一种抚慰?又或者,来...

Android的LevelListDrawable简单使用

前言LevelListDrawable是通过改变层级值来显示对应的图片,除了下面的开关灯,还有WiFi的状态显示,电池状态的显示也可以用这种。在公司好像没发现有人使用过LevelListDrawable(或者我看代码太少了哈)。自己懒得写了,摘抄一些网友写的,以便自己学习。以下内容都是摘抄...

设计模式:装饰模式或包装模式

什么是设计模式装饰模式又名包装模式。装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案。有透明和半透明两种,大部分都是半透明的,半透明的装饰模式是介于装饰模式和适配器模式之间的。装饰模式的核心:功能扩展。透明和半透明的区别:透明的装饰模式,要求具体构件角色、装饰角色的接...

Service的Context介绍

前言对于Application,Activity和Service这几个类,我们是很[熟悉]的。确实[熟悉],作为App开发这基本都要面对这几个类。几天记录一下Service的Context创建,其实也就是简单的说说。正文Context的使用场景使用Context调用方法,比如启动A...

[NDK开发]Android JNI 中新增JNI层日志打印

前言在上一篇的基础上《[NDK开发]Android JNI 开发之第一个 JNI 实例》,进行新增log打印正文Android.mk添加LOCAL_LDLIBS := -llog完整代码如下LOCAL_PATH := $(call my-dir)include $(CLEAR...

getMimeTypeFromExtension记录

前言在Android 13的MediaProvider中有涉及到getMimeTypeFromExtension()的使用,主要是获取文件的MimeType属性。记录于此,方便自己差异。下面代码来自MediaProvider正文获取mimeType//获取文件后缀扩展String...