预告
后续可能会推更一个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加载一个本地库时,虚拟机会在本地库中查找下述的导出的程序入口:
此时我们可以将JavaVM缓存下来,供以后获取JNIEnv使用。
下面是在任何位置获取JNIEnv的例子:
不要混淆ID和引用
JNI将对象作为引用。类,字符串和数组是特殊类型的引用。JNI将方法和字段作为ID。一个ID不是一个参考。不要将类引用称为“类ID”,也不要将方法ID称为“方法引用”。
引用是可以由本地代码显式管理的虚拟机资源。例如,JNI函数DeleteLocalRef允许本地代码删除本地引用。相比之下,字段和方法ID由虚拟机管理并保持有效直到其定义的类被卸载。在虚拟机卸载定义的类之前,本机代码不能显式删除字段或方法ID。
缓存字段和方法ID
本地代码通过将字段或方法的名称和类型描述符指定为字符串然后从虚拟机获取字段或方法ID。使用名称和类型字符串的字段和方法查找速度很慢。缓存这些ID通常是有利的,未能缓存字段和方法ID是本机代码中的常见性能问题。
缓存字段或方法ID建议使用类的静态代码块的方式进行缓存。
避免过量创建本地引用
虽然说本地引用会在函数结束时自动释放,但是JNI对于本地引用的个数是有一定的限制的,一般是限制到512个,因此需要注意一些调用链比较长的函数或者是在循环体内返回的本地引用在使用完毕后及时进行释放,以保证GC的正常工作和内存的稳定。
NDK错误定位
在开发的过程中经常会出现一些Native层的崩溃,然后在Logcat中又没有显示具体位置的,这时候可以使用NDK中的工具包进行定位。
addr2line的命令使用方式如下:
其中-C -f表示打印错误行数所在的函数名称,-e表示打印错误地址的对应路径及行数。注意不同的CPU架构需要使用不同的addr2line,比如mac系统的addr2line就存在于
那么怎么通过Logcat定位到崩溃的内存地址呢?例如有以下崩溃日志:
那么所在的下一行就是崩溃的内存地址,也就是说上面崩溃日志的错误地址是。
参考资料
《JNI编程指南与规范》
领取专属 10元无门槛券
私享最新 技术干货