JNI-动态库的函数注册

时光荏苒,距离上次的文章已经3年多了。那这3年干啥了呢?平凡生活,在人间凑数。

来吧,展示!show me you code

在Linux平台下so库分为动态库和静态库。表现形式以.so为后缀动态库和.a为后缀的静态库。关于这2个的有关知识这里不过多介绍,不是我们今天的重点。

在动态库里函数注册分为2种:静态注册和动态注册。

静态注册

静态注册,你工作如果是与jni打交道。我们平时写的,写一个jni方法和它对应的头文件(方法名,参数进行对应)。调用的时候,这个调用的过程就是静态注册。
但是这个调用的过程有一个查找(根据方法名,方法唯一签名)的步骤,这个查找的过程就是耗时的过程。因此有了动态注册。

先看一下平时写jni的静态注册,方法名必须是Java_java类全名_方法名,java类全名的._替换,不过目前的as可以自动生成,nice。

比如:

// JNIMethodDynamic.java
package com.bj.gxz.jniapp;
public class JNIMethodDynamic {// Used to load the 'native-lib' library on application startup.static {System.loadLibrary("native-lib");}/*** A native method that is implemented by the 'native-lib' native library,* which is packaged with this application.*/public native String stringFromJNI();public native int sum(int x, int y);
}
// jni_method_dynamic.cpp
#if 0
extern "C" JNIEXPORT jstring JNICALL
Java_com_bj_gxz_jniapp_JNIMethodDynamic_stringFromJNI(JNIEnv *env,jobject thiz) {std::string hello = "Hello from C++";return env->NewStringUTF(hello.c_str());
}extern "C"
JNIEXPORT jint JNICALL
Java_com_bj_gxz_jniapp_JNIMethodDynamic_sum(JNIEnv *env, jobject thiz, jint x, jint y) {return x + y;
}
#else
#endif

动态注册

动态注册,在jni中提供了这种机制。在<jni.h>这个头文件中提供了如下2个方法声明。

/** Prototypes for functions exported by loadable shared libs.  These are* called by JNI, not provided by JNI.*/
JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved);
JNIEXPORT void JNI_OnUnload(JavaVM* vm, void* reserved);

注释大概意思就是:当系统加载动态库的时候(在java中就是System.loadLibrary())由JNI调用。
如何实现呢? 在jni中我们可以复写这2个方法,有点类似java语言的子类复写父类的方法的感觉。嘿嘿,什么都是靠感觉。
当然一些初始化的工作你也可以在这里来做。

复写第一个方法可以把我们自己的方法注册到系统中去,然后在jni层直接调用即可,不用按照静态注册的规则来写了。
复写第二个方法可以把我们自己的方法/jni进行解注册

如何注册呢 ? 主要是jni头文件的

jint RegisterNatives(jclass clazz, const JNINativeMethod* methods,jint nMethods) 参数1:对应java类的native的方法声明的类参数2:JNINativeMethod结构体数组,java类native方法和jni方法的映射关系参数3:注册了的方法的个数

主要是参数2,它的结构体如下:

typedef struct {const char* name;   //java类native方法的名字,不带包的路径const char* signature; //java类native方法的签名,字符串表示void*       fnPtr;//void*类型的函数指针,映射到jni层的方法名字
} JNINativeMethod;

ok,我们把常规的写法(静态注册)的stringFromJNI和sum进行改造,声明一个JNINativeMethod的结构体数组并进行映射。

JNINativeMethod gMethods[] = {{"stringFromJNI", "()Ljava/lang/String;", (void *) getString},{"sum",           "(II)I",                (void *) add}
};

关于方法的签名都是固定的规则,如果你怕写错也可以用javap -s class文件命令查看,如下:

getString和add方法具体实现为:

jstring getString(JNIEnv *env, jobject thiz) {return env->NewStringUTF("Hello from C++");
}
jint add(JNIEnv *env, jobject thiz, jint x, jint y) {return x + y;
}

嗯,至此我们完成了第2个参数。第3个参数我们动态注册了2个方法传2即可。 我们再来看第一个参数jclass。
如果获取jclass呢,就是通过java_cls = env->FindClass(JAVA_CLASS)常见的这个方法即可JAVA_CLASS就是在java中声明native方法类的全名,我的demo里是:com/bj/gxz/jniapp/JNIMethodDynamic

那JNIEnv *env怎么获取呢?

备注:因为我们没有从头开始将jni的一些知识,这里简单说一些JNIEnv *env,它是一个结构体指针,它指向可用JNI函数表的接口指针,提供了一些JNI系统函数

JNI_OnLoad的系统方法中的第一个参数中JavaVM *vm可以获取到JNIEnv,

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {JNIEnv *env;if (vm->GetEnv(reinterpret_cast<void **>(&env), JNI_VERSION_1_6) != JNI_OK) {return JNI_ERR;}// ...
}

到此,注册已经完成了。
如何解注册呢 ?
复写系统的JNI_OnUnload方法,调用UnregisterNatives即可。

JNIEXPORT void JNI_OnUnload(JavaVM *vm, void *reserved) {JNIEnv *env;if (vm->GetEnv(reinterpret_cast<void **>(&env), JNI_VERSION_1_6) != JNI_OK) {return;}jclass java_cls = env->FindClass(JAVA_CLASS);if (java_cls != nullptr) {jint ret = env->UnregisterNatives(java_cls);LOG_D("JNI_OnUnload Call ret=%d=", ret);}
}

完事,我们写个demo测试一下:

public class MainActivity extends AppCompatActivity {private JNIMethodDynamic jniMethodDynamic = new JNIMethodDynamic();@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);TextView tv = findViewById(R.id.sample_text);tv.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {Log.d("jni", "stringFromJNI:" + jniMethodDynamic.stringFromJNI());Log.d("jni", "sum:" + jniMethodDynamic.sum(1, 2));}});}
}

输出

09-19 05:26:51.986 2106-2106/com.bj.gxz.jniapp D/JNI: JNI_OnLoad Call ret=1
09-19 05:27:12.590 2106-2106/com.bj.gxz.jniapp D/JNI: stringFromJNI:Hello from C++
09-19 05:27:12.590 2106-2106/com.bj.gxz.jniapp D/JNI: sum:3

代码地址
https://github.com/ta893115871/JNIAPP

Android-JNI开发系列《一》-动态库的函数注册相关推荐

  1. Android JNI开发系列(二)HelloWorld

    2019独角兽企业重金招聘Python工程师标准>>> 入门HelloWorld 新建项目 Configure your new project部分选中 Include C++ Su ...

  2. android 字符串函数,Android JNI开发系列(六)字符串操作

    JNI字符串操作 字符串是引用数据类型,不属于基本数据类型 Java 使用unicode编码,C使用UTF-8,所以在操作中 C语言的字符串操作在头文件中 示例代码 public native Str ...

  3. 【Android FFMPEG 开发】Android Studio 工程配置 FFMPEG ( 动态库打包 | 头文件与函数库拷贝 | CMake 脚本配置 )

    文章目录 I . FFMPEG 交叉编译后的函数库及头文件 II . FFMPEG 静态库打包动态库 ( 仅做参考 ) III . 创建 Android Studio 工程 IV . FFMPEG 头 ...

  4. 【Android NDK 开发】Android Studio 使用 CMake 导入动态库 ( 构建脚本路径配置 | 指定动态库查找路径 | 链接动态库 )

    文章目录 I . CMake 引入动态库与静态库区别 II . Android Studio 中 CMake 引入动态库流程 III . 指定动态库查找路径 IV . 链接函数库 V . 完整代码示例 ...

  5. 【Android 安装包优化】使用 lib7zr.so 动态库处理压缩文件 ( 修改 7zr 交叉编译脚本 Android.mk | 交叉编译 lib7zr.so 动态库 )

    文章目录 一.修改 7zr 交叉编译脚本 Android.mk 二.完整的 7zr 交叉编译脚本 Android.mk 三.交叉编译 lib7zr.so 动态库 四.参考资料 一.修改 7zr 交叉编 ...

  6. Android JNI开发入门之二

    在上一篇文章<Android JNI开发入门之一>中,我介绍了Android应用程序(APK)怎样通过JNI调用Native C实现的共享库.本文将进一步介绍Android应用程序通过JN ...

  7. Android自定义控件开发系列(零)——基础原理篇

    在后边的文章中发现在说Android自定义时,有时候要重复解释很多东西,所以想想返回来增加一篇"基础原理篇",直接进入正题吧-- 首先的问题是:在Android项目开发中,什么时候 ...

  8. Android蓝牙开发系列文章-扫不到蓝牙设备,你的姿势对了吗?

    在写<Android蓝牙开发系列文章-蓝牙音箱连接>时,计划细化出两篇文章,分别是: 关于蓝牙设备类型分类的,这个已经完成了,阅读请点击<Android蓝牙开发系列文章-蓝牙设备类型 ...

  9. 【错误记录】NDK 导入外部 so 动态库报错 ( java.lang.UnsatisfiedLinkError | Android Studio 配置外部 so 动态库两种方法 )

    文章目录 一.报错信息 二.解决方案 ( Android Studio 配置外部 so 动态库两种方法 ) 1.jniLibs 目录存放 2.libs 目录存放 一.报错信息 外部引用 so 动态库 ...

最新文章

  1. Cesium 创建Geometry
  2. ML:MLOps系列讲解之《端到端 ML工作流生命周期》解读
  3. python实现k core算法_Python core.take方法代码示例
  4. 『ACM C++』 PTA 天梯赛练习集L1 | 016-017
  5. P2824-[HEOI2016/TJOI2016]排序【线段树,二分】
  6. egg框架访问 Mysql 数据库 egg-mysql 增删改查
  7. 【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记30 ScrollView Demo实战
  8. 服务器怎么控制忽略样式_看问题要看到本质:从Web服务器说起
  9. 思维导图—Git命令全集
  10. 安卓软件开发面试题!五年Android开发者小米、阿里面经,小白也能看明白
  11. 服务器网站不用80端口,云服务器80端口不用备案
  12. 关于JAVA的优势——跨平台和健壮性
  13. ASP网页HTTP 错误 404.3 - Not Found解决方案
  14. .Bear勒索病毒如何删除它 .Bear后缀文件如何恢复(Dharma家族)
  15. 如何修改安卓应用图标和程序名称
  16. 关于uC/OS-II 概述
  17. 中国线上超市行业营销态势与投资盈利预测报告(2022-2027)
  18. 初出茅庐的第一篇文章
  19. ORB-SLAM2系列第一章——简介
  20. 图解 Word2Vec

热门文章

  1. PIE SDK矢量点生成等值线、面
  2. 滴滴 Web 移动端组件库 cube-ui 开源
  3. 李洪强和你一起学习前端之(9)规避脱标,CSS可见性,滑动门案例
  4. JVM性能优化, Part 5:Java的伸缩性
  5. python3学习之元组
  6. 采用Visual Stuidio 2010 创建网站栏
  7. 远离你的电脑,代码写得更好!
  8. OpenCV for Android开发环境Win7平台搭建(转)
  9. 看上冰岛的域名con.is
  10. HTML和CSS是什么玩意儿