前言
之前其实写过,代码不见了,为了走一下流程,重新简单的写了一个。
正文
动态注册
动态注册java的Native方法,使得c/c++方法名可以和java的Native方法名可以不同。动态注册是将二者方法名关联起来,以后在修改Native方法名时,只需修改动态注册关联的方法名称即可。
-
优点: 灵活性高, 更改类名,包名或方法时, 只需对更改模块进行少量修改, 效率高
-
缺点: 对新手来说稍微有点难理解, 同时会由于搞错签名, 方法, 导致注册失败
实战
定义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(签名),需要通过一定的方式获取。
两种方式多需要先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配置跟静态注册一样,这里就懒得重复了。
可以看最近整理的《》
参考文章
-
《》
-
《》
-
《》
-
《》
历史上的今天
暂无评论...
随机推荐
白岩松:幸福在哪里
一走在人群中,我习惯看一看周围人的手腕,那里似乎藏着一个属于当代中国人的内心秘密,从不言说,却日益增多。越来越多的人,不分男女,会戴上一个手串,这其中,不乏有人仅仅是为了装饰;更多的却带有祈福与安心的意味,这手串停留在装饰与信仰之间,或左或右。这其中,是怎样的一种相信或怎样的一种抚慰?又或者,来...
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...