Android Studio NDK JNI动态注册本地方法
概述
可能大家觉得javah生成的函数名又臭又长,不太好看。这里可以提供另外一种方法来动态注册c++函数,让其根Java中的native方法关联起来。
实现
这里通过JNIEnv的Resisternatives方法完成方法的注册。相关方法介绍:
//方法映射描述结构体 typedef struct { const char* name;//Java方法名 const char* signature;//方法签名 void* fnPtr;//C++ 方法指针 } JNINativeMethod; //这是JNIEnv提供的注册本地方法 //clazz:方法对应的class //methods:对应的方法数组指针 //nMethods:有几个方法 //返回值:注册成功返回JNI_OK jint RegisterNatives(jclass clazz, const JNINativeMethod* methods, jint nMethods); //当本地库被加载时VM调用JNI_OnLoad(例如,通过系统调用LoadLibrary)。JNI_OnLoad必须返回由本地库所需的JNI版本。 //为了使用任何新的JNI函数,一个本地库必须导出JNI_OnLoad函数并返回JNI_VERSION_1_2或更高的版本。 //如果本地库不导出JNI_OnLoad功能,VM假定库只需要JNI_VERSION_1_1版本。 //如果虚拟机不认JNI_OnLoad返回的版本号,本地库不能加载。 JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved);
看了上面的函数,我们可以知道在loadLibrary的时候会首先调用JNI_OnLoad。因此打算在JNI_OnLoad中完成方法注册:
/filename:my.cpp // Created by wastrel on 2016/9/8. // #include <stddef.h> #include "jni.h" //返回一个字符串 JNIEXPORT jstring JNICALL native_hello(JNIEnv *env, jclass clazz) { return env->NewStringUTF("Hello from C++"); } //求两个int的值 JNIEXPORT jint JNICALL native_add(JNIEnv *env, jobject object, jint a, jint b) { return a + b; } //方法数组,JNINativeMethod的第一个参数是Java中的方法名,第二个参数是函数签名,第三个参数是对应的方法指针。 //Java方法的签名一定要与对应的C++方法参数类型一致,否则注册方法可能失败。 static JNINativeMethod method_table[] = { {"native_hello", "()Ljava/lang/String;", (void *) native_hello}, {"native_add", "(II)I", (void *) native_add} }; JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved) { //OnLoad方法是没有JNIEnv参数的,需要通过vm获取。 JNIEnv *env = NULL; if (vm->AttachCurrentThread(&env, NULL) == JNI_OK) { //获取对应声明native方法的Java类 jclass clazz = env->FindClass("com/example/registerjni/HelloJNI"); if (clazz == NULL) { return JNI_FALSE; } //注册方法,成功返回正确的JNIVERSION。 if (env->RegisterNatives(clazz, method_table, sizeof(method_table)/ sizeof(method_table[0]))==JNI_OK) { return JNI_VERSION_1_4; } } return JNI_FALSE; }
对应的native声明Java文件:
package com.example.registerjni; /** * Created by wastrel on 2016/9/8. */ public class HelloJNI { static { System.loadLibrary("helloJNI"); } public native static String native_hello(); public native int native_add(int a,int b); }
注意:如果你的方法声明了static,那么对应的第二个参数应该是jclass类型。如果你清楚他的实际类型,即便你写成jobject也不会引起程序错误,因为jclass本身也是jobject。但还是建议写为正确的方式,这样可以显得清晰一些。
注意:C++和Java有所不同,如果把static JNINativeMethod method_table 写在开头,你编译的时候会提示找不到函数指针,这是因为自上而下编译的原因。所以应该把定义写在实现方法后面,或者用一个头文件来完成函数的定义。
后记
这只是一种建立Native方法与Java方法的方式,如果没有特殊的需要,不建议使用这种方法来注册。因为使用这种优雅的注册方法,你必须确保你的函数方法和签名的正确性。这可能会增加出错的概率。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 周边上新:园子的第一款马克杯温暖上架
2015-03-27 在老项目中使用Gradle:更改默认目录结构