JNI之常见技巧与陷阱

简介: NDK/JNI连载系列

预告

后续可能会推更一个FFmpeg系列的入门博客,大概涉及到FFmpeg解封装、FFmpeg编解码、FFmpeg进行音频重采样、使用FFMpeg将mp3转码成aac、使用FFmpeg合并拼接音视频等。

另外如果有时间可能也会更新几篇关于ffplay的文章,敬请关注。

本文将作为JNI系列的一个结尾,下面是笔者在学习使用JNI的所记录的一些笔记与技巧。

JNIEnv的线程限制

一个JNIEnv指针仅在其相关联的线程中有效。你不能将这个指针从一个线程中传递给另一个线程,或者在多线程中缓存和使用它。Java虚拟机在同一个线程的连续调用中传递给本地方法相同的JNIEnv指针,但是从不同线程中调用本地方法时传递的是不同的JNIEnv指针。应当避免缓存一个线程的JNIEnv指针并在另一个线程中使用指针的常见错误。

本地引用(局部引用)仅在创建它的线程中有效。你不能将本地引用从一个线程中传递到另一个线程。每当有多个线程可能使用相同引用的可能性时,应始终将本地引用转换为全局引用。

JNIEnv是用作线程局部存储。因此,使用者不能在多线程间共享一个JNIEnv变量。如果在一段代码中没有其它办法获得它的JNIEnv,使用者可以共享JavaVM对象,使用GetEnv来取得该线程下的JNIEnv。

如果你使用AttachCurrentThread连接(attach)了Native进程,正在运行的代码在线程分离(detach)之前绝不会自动释放局部引用。使用者创建的任何局部引用必须手动删除。通常,任何在循环中创建局部引用的Native代码可能都需要做一些手动删除。

全局获取JNIEnv

一个JNIEnv指针仅在与其相关联的线程中有效。对于本地方法,这通常不是问题,因为他们从虚拟机接受JNIEnv指针作为第一个参数。然而有时候可能不需要直接从虚拟机调用的本地代码来获取属于当前线程的JNIEnv接口指针。例如通过JNI在Native开启了一个子线程处理某些任务,在这些任务处理完毕后需要将处理结果回调给java层。
这种情况可以通过缓存JavaVM获取当前线程的JNIEnv然后进行java方法的回调。

当System加载一个本地库时,虚拟机会在本地库中查找下述的导出的程序入口:

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved);

此时我们可以将JavaVM缓存下来,供以后获取JNIEnv使用。

下面是在任何位置获取JNIEnv的例子:

JavaVM *globalJVM = nullptr;
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved) {
    globalJVM = jvm;
    return JNI_VERSION_1_6;
}

JNIEnv *getCurrentEnv(int *attach) {
    if (globalJVM == nullptr) return nullptr;
    *attach = 0;
    JNIEnv *jni_env = nullptr;
    int result = globalJVM->GetEnv((void **) &jni_env, JNI_VERSION_1_6);
    if (result == JNI_EDETACHED || jni_env == nullptr) {
        result = globalJVM->AttachCurrentThread(&jni_env, nullptr);
        if (result < 0) {
            jni_env = nullptr;
        } else {
            *attach = 1;
        }
    }
    return jni_env;
}

不要混淆ID和引用

JNI将对象作为引用。类,字符串和数组是特殊类型的引用。JNI将方法和字段作为ID。一个ID不是一个参考。不要将类引用称为“类ID”,也不要将方法ID称为“方法引用”。

引用是可以由本地代码显式管理的虚拟机资源。例如,JNI函数DeleteLocalRef允许本地代码删除本地引用。相比之下,字段和方法ID由虚拟机管理并保持有效直到其定义的类被卸载。在虚拟机卸载定义的类之前,本机代码不能显式删除字段或方法ID。

缓存字段和方法ID

本地代码通过将字段或方法的名称和类型描述符指定为字符串然后从虚拟机获取字段或方法ID。使用名称和类型字符串的字段和方法查找速度很慢。缓存这些ID通常是有利的,未能缓存字段和方法ID是本机代码中的常见性能问题。

缓存字段或方法ID建议使用类的静态代码块的方式进行缓存。

避免过量创建本地引用

虽然说本地引用会在函数结束时自动释放,但是JNI对于本地引用的个数是有一定的限制的,一般是限制到512个,因此需要注意一些调用链比较长的函数或者是在循环体内返回的本地引用在使用完毕后及时进行释放,以保证GC的正常工作和内存的稳定。

NDK错误定位

在开发的过程中经常会出现一些Native层的崩溃,然后在Logcat中又没有显示具体位置的,这时候可以使用NDK中的addr2line工具包进行定位。

addr2line的命令使用方式如下:

addr2line的绝对路径 -C -f -e so文件的绝对路径  错误内存地址

其中-C -f表示打印错误行数所在的函数名称,-e表示打印错误地址的对应路径及行数。
注意不同的CPU架构需要使用不同的addr2line,比如mac系统的addr2line就存在于ndk目录/toolchains/llvm/prebuilt/darwin-x86_64/bin

那么怎么通过Logcat定位到崩溃的内存地址呢?例如有以下崩溃日志:

2022-03-29 22:28:58.462 22761-22761/? A/DEBUG: ABI: 'arm64'
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG: Timestamp: 2022-03-29 22:28:58+0800
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG: pid: 22652, tid: 22733, name: Thread-2  >>> com.fly.jnitest <<<
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG: uid: 10147
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG: signal 5 (SIGTRAP), code 1 (TRAP_BRKPT), fault addr 0x7984411f40
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x0  00000079de54d200  x1  0000007a73fd41c0  x2  0000000000000000  x3  00000079eec9fcda
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x4  00000079837e5c08  x5  00000079eead6059  x6  0000000000000001  x7  00000079837e5838
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x8  0000000000000000  x9  b4175a2f4989bf75  x10 0000000000430000  x11 0000000000000001
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x12 0000000000000000  x13 0000000000000000  x14 0000000000000012  x15 00000000000000ff
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x16 00000079ef1c8748  x17 0000007a73c62350  x18 00000079802a6000  x19 00000079837e5d50
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x20 0000007a700ee0dc  x21 00000079837e5d50  x22 0000587c0000587c  x23 00000079837e5dd8
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x24 00000079837e5d50  x25 00000079837e5d50  x26 00000079837e6020  x27 0000007a74148020
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     x28 0000007ffad6a430  x29 00000079837e5cf0
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG:     sp  00000079837e5ce0  lr  0000007984411f3c  pc  0000007984411f40
2022-03-29 22:28:58.462 22761-22761/? A/DEBUG: backtrace:
2022-03-29 22:28:58.463 22761-22761/? A/DEBUG:       #00 pc 0000000000000f40  /data/app/com.fly.jnitest-X01T6VOuYKufX3tBWVg2vA==/lib/arm64/libjnitest.so (test(void*)+24) (BuildId: f06b5f684113a965be07abbcf0bb4e5488d31870)
2022-03-29 22:28:58.463 22761-22761/? A/DEBUG:       #01 pc 00000000000e1100  /apex/com.android.runtime/lib64/bionic/libc.so (__pthread_start(void*)+36) (BuildId: c042ffb4e195c9462700c20f99189c2b)
2022-03-29 22:28:58.463 22761-22761/? A/DEBUG:       #02 pc 0000000000083ab0  /apex/com.android.runtime/lib64/bionic/libc.so (__start_thread+64) (BuildId: c042ffb4e195c9462700c20f99189c2b)

那么 backtrace:所在的下一行就是崩溃的内存地址,也就是说上面崩溃日志的错误地址是0000000000000f40

参考资料

《JNI编程指南与规范》

推荐阅读

JNI基础简介
JNI之数组与字符串的使用
JNI之动态注册与静态注册
JNI之访问java属性和方法
JNI之缓存与引用
JNI之异常处理

关注我,一起进步,人生不止coding!!!

目录
相关文章
|
8月前
|
存储 Java C++
[NDK/JNI系列02] JNI的设计原理与数据类型
[NDK/JNI系列02] JNI的设计原理与数据类型
57 0
[NDK/JNI系列02] JNI的设计原理与数据类型
|
2月前
|
存储 JavaScript 前端开发
如何优化代码以避免闭包引起的内存泄露
本文介绍了闭包引起内存泄露的原因,并提供了几种优化代码的策略,帮助开发者有效避免内存泄露问题,提升应用性能。
|
7月前
|
Java 编译器 Android开发
一篇文章讲明白jni中arm64
一篇文章讲明白jni中arm64
92 0
|
8月前
|
存储 架构师 Linux
内存泄漏专题(7)hook之宏定义
内存泄漏专题(7)hook之宏定义
98 0
|
安全 C语言
C语言编程陷阱:库函数陷阱
会造成较高的系统负担 暂存然后以大块写入的方式 缓冲数组最好时成为静态数组,或者显示内存申请
65 1
|
8月前
|
监控 Java Unix
日常知识点之内存泄露定位手段(c语言hook malloc相关方式)
日常知识点之内存泄露定位手段(c语言hook malloc相关方式)
161 0
|
缓存 运维 Java
JNI异常处理
NDK/JNI连载系列
89 0
|
IDE Java 开发工具
JNI的开发方法
本文通过一个案例,教读者一步一步实现一个简单的JNI项目。
158 0
JNI的开发方法
|
Java Unix Linux
JNI学习(0)——关于JNI
JNI学习(0)——关于JNI
150 0
JNI学习(0)——关于JNI
|
Java
JNI学习(1)——生成对应的头文件
JNI学习(1)——生成对应的头文件
362 0
JNI学习(1)——生成对应的头文件